为了账号安全,请及时绑定邮箱和手机立即绑定

没有学不会的C++:避免在构造函数或析构函数中调用虚函数

标签:
C++

我们先来看下下面程序

#include <iostream>
using namespace std;
                                            
class Dog {
public:
    Dog () { cout << "I'm a dog" << endl; }
    virtual ~Dog () { cout << "destroy a dog" << endl; }

	void bark() { cout << "dog bark" << endl; }
    void seeDog() { bark(); }
};

class YellowDog : public Dog {
public:
    YellowDog() { cout << "I'm a yellow dog" << endl; }
    virtual ~YellowDog() { cout << "destroy a yellow dog" << endl;}

    void bark() { cout << "yellow dog bark" << endl; }
};


int main() {
    YellowDog yd = YellowDog();
    yd.seeDog();

    return 0;
}

运行上述程序后,得到以下输出:

I'm a dog
I'm a yellow dog
dog bark
destroy a yellow dog
destroy a dog

可以看到在创建 YellowDog 对象时,首先会调用基类的构造函数,而且虽然是 YellowDog 调用的 seeDog,但执行的却是 Dog::bark,此时,如果你想要调用 YellowDog::bark,你需要把基类 Dog 中的 bark 声明为 virtual 类型,如下:

	virtual void bark() { cout << "dog bark" << endl; }

同时,为了更好的可读性,建议你也将子类中的继承自基类的虚函数也显示的声明为 virtual,此时你再运行上面的程序后,就可以看到 YellowDog::bark 被调用了,这就是 C++ 中的多态或动态绑定。

I'm a dog
I'm a yellow dog
yellow dog bark
destroy a yellow dog
destroy a dog

下面,我们做点小改动,故意让 C++ 中的动态绑定失效,我们把 seeDog() 中的 bark() 挪到 Dog 的构造函数中,看下会出现什么情况

    // ...
    Dog () { cout << "I'm a dog" << endl; bark(); }
    // ...

此时输出变成了

I'm a dog
dog bark
I'm a yellow dog
destroy a yellow dog
destroy a dog

可以看到虽然声明了虚函数,但却没有进行动态绑定,原因在于在调用 Dog 的构造函数时,YellowDog 对象还没有被构造完成,对一个不存在的对象调用其成员函数是非常危险的,所以编译器在这里选择了调用 Dog::bark,而不是 YellowDog::bark,同时要注意的是,在构造函数中尽可能的只做最简单的初始化操作,避免复杂的函数调用。

现在我们把 bark() 放在 Dog 中的析构函数中,看下会发生什么情况

    // ...
    Dog () { cout << "I'm a dog" << endl;  }
    virtual ~Dog () { cout << "destroy a dog" << endl; bark();}
    // ...

输出如下

I'm a dog
I'm a yellow dog
destroy a yellow dog
destroy a dog
dog bark

再一次,多态的特性没有生效,原因在于在析构 YellowDog 的对象时,先调用 YellowDog 的析构函数,即在调用 bark 时,YellowDog 部分的数据已经被清理,此时再调用该对象的虚函数也是非常危险的,所以为了安全起见,编译器又一次选择了调用基类的 bark

以上,我们学到了一条宝贵的经验:

不要在构造函数或析构函数中调用虚函数

点击查看更多内容
1人点赞

若觉得本文不错,就分享一下吧!

评论

作者其他优质文章

正在加载中
感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦
今天注册有机会得

100积分直接送

付费专栏免费学

大额优惠券免费领

立即参与 放弃机会
意见反馈 帮助中心 APP下载
官方微信

举报

0/150
提交
取消