禁止在构造函数里调用虚函数

简介: 禁止在构造函数里调用虚函数

在构造函数中调用虚函数会导致程序出现莫名其妙的行为,这主要是对象还没有完全构造完成。下面我们先来看一段代码:

class B
{
    protected B()
    {
        Method();
    }
    protected virtual void Method()
    {
        Console.WriteLine("B Method");
    }
}
class A:B
{
    private readonly string str = "你好";
    public A(string str)
    {
        this.str=str;
    }
    protected override void Method()
    {
        Console.WriteLine(str);
    }
    public static void Main()
    {
        var d = new A("A Method");
    }
}

在这里我要问一下各位读者,在上述代码中打印出的内容是什么?大部分读者会回答 “A Method” ,实际上的答案是 “你好” 。这是为什么呢?这是因为基类的构造函数调用一个定义在本类中的但是为派生类所重写的虚函数,程序运行的时候会调用派生类的版本,程序在运行期的类型是 A 而不是 B。在 C# 中系统会认为这个对象是一个可以正常使用的对象,这是因为程序在进入构造函数的函数体之前已经把该对象的所有成员变量都进行了初始化。但是者并不意味着这些成员变量的值和开发人员最终想要的值相符,因为程序仅仅执行了成员变量的初始化语句,而没有执行构造函数中的逻辑。

我们将前面的代码稍加修改看一下:

abstract class B
{
    protected B()
    {
        Method();
    }
    protected abstract void Method();
}
class A:B
{
    private readonly string str = "你好";
    public A(string str)
    {
        this.str=str;
    }
    protected override void Method()
    {
        Console.WriteLine(str);
    }
    public static void Main()
    {
        var d = new A("A Method");
    }
}

针对上述代码,我想问问各位读者,这段代码可以通过编译码?答案是可以通过编译,这是因为程序就不会创建一个类型为 B 的对象,他创建的对象只是 B 的实现了 Method 方法的子类,程序代码所运行的也是那个子类的 Method 方法。这么做主要是为了避免在构造函数中调用抽象类中的方法,防止抛出异常。虽然这么写可以避免这个问题但是还存在一个很大的缺陷,它会造成 str 这个对象在整个生命周期中无法保持恒定的值。在构造函数还没有把该对象初始化完成之前,它的取值是由初始化语句决定的,但是执行完构造函数之后它的值却变成了构造函数中所设定的那个值。派生类对象所具备的成员变量的默认值是由初始化语句或者系统来确定的,因此开发人员如果想要在构造函数中给这些变量赋值那么就必须等到程序运行到构造函数时才可以。


Tip:C# 对象的运行期类型是一开始就定好的,即便基类是抽象类也依然可以调用其中的虚方法。

小结

在基类构造函数中调用虚函数会导致代码严重依赖于派生类的实现,然后这些实现是无法控制且容易出错的。如果要避免错误,派生类就必须通过初始化语句把所有的实例变量设置好,但是这又会使得开发人员无法运用更多的编程技巧。也就是说在这种情况下派生类必须定义默认构造函数,并且不能定义别的构造函数,这将会给开发人员带来很大的负担。


目录
相关文章
|
网络协议 安全 网络安全
图解OSI七层模型,2024最强科普!
【7月更文挑战第20天】
2983 2
图解OSI七层模型,2024最强科普!
|
算法 API 图形学
【Qt 学习笔记】QWidget的geometry属性及window frame的影响
【Qt 学习笔记】QWidget的geometry属性及window frame的影响
921 2
|
运维 网络协议 Linux
Linux与Windows下追踪网络路由:traceroute、tracepath与tracert命令详解
Linux与Windows下追踪网络路由:traceroute、tracepath与tracert命令详解
2889 0
|
网络安全 数据安全/隐私保护
解决git@github.com: Permission denied (publickey). fatal: Could not read from remote repository. Pleas
解决git@github.com: Permission denied (publickey). fatal: Could not read from remote repository. Pleas
2684 0
|
编解码 开发框架 监控
Qt搭建RTSP服务器
一、项目背景 随着物联网技术不断发展,视频监控系统在各个领域的应用越来越广泛。其中,RTSP(Real Time Streaming Protocol)是一种常用的流媒体传输协议,可以实现对实时音视频数据的传输和播放。为了实现视频监控系统的网络化和智能化,需要开发一个基于RTSP协议的视频流服务器,能够接收前端设备的视频流,并提供RTSP协议的服务,方便客户端进行实时的视频浏览、回放等操作。 在开发过程中,为了提高开发效率、减少开发难度和成本,同时具备良好的可扩展性和可维护性,我选择使用Qt和Live555库来搭建RTSP服务器。Qt是一个跨平台的C++应用程序开发框架,具有完善的GUI界
508 0
|
6天前
|
弹性计算 人工智能 安全
云上十五年——「弹性计算十五周年」系列客户故事(第二期)
阿里云弹性计算十五年深耕,以第九代ECS g9i实例引领算力革新。携手海尔三翼鸟、小鹏汽车、微帧科技等企业,实现性能跃升与成本优化,赋能AI、物联网、智能驾驶等前沿场景,共绘云端增长新图景。
|
12天前
|
存储 弹性计算 人工智能
【2025云栖精华内容】 打造持续领先,全球覆盖的澎湃算力底座——通用计算产品发布与行业实践专场回顾
2025年9月24日,阿里云弹性计算团队多位产品、技术专家及服务器团队技术专家共同在【2025云栖大会】现场带来了《通用计算产品发布与行业实践》的专场论坛,本论坛聚焦弹性计算多款通用算力产品发布。同时,ECS云服务器安全能力、资源售卖模式、计算AI助手等用户体验关键环节也宣布升级,让用云更简单、更智能。海尔三翼鸟云服务负责人刘建锋先生作为特邀嘉宾,莅临现场分享了关于阿里云ECS g9i推动AIoT平台的场景落地实践。
【2025云栖精华内容】 打造持续领先,全球覆盖的澎湃算力底座——通用计算产品发布与行业实践专场回顾
|
3天前
|
云安全 人工智能 安全
Dify平台集成阿里云AI安全护栏,构建AI Runtime安全防线
阿里云 AI 安全护栏加入Dify平台,打造可信赖的 AI