先看一段示例代码。
#include <iostream>
using namespace std;
class Base
{
public:
Base()
{
cout << "Base constructor" << endl;
cout << "Value is " << GetValue() << endl;
}
~Base()
{
cout << "Base destructor" << endl;
cout << "Value is " << GetValue() << endl;
}
protected:
virtual int GetValue() { return 1; }
};
class Derive : public Base
{
public:
Derive() : Base()
{
cout << "Derive constructor" << endl;
}
~Derive()
{
cout << "Derive destructor" << endl;
}
protected:
virtual int GetValue() { return 2; }
};
int main()
{
Derive derive;
return 0;
}
这段示例代码,类Derive派生于Base,Base的构造函数与虚构函数均调用虚函数GetValue(),根据C++多态特性,应该是要调用Derive的GetValue()返回2,真的是这样吗?让我们看下最终的输出。
从输出的结果看,Base的构造函数与虚构函数均调用Base的GetValue()返回1,而不是Derive的GetValue()。从汇编代码进一步确认,C++编译器确实是直接调用Base的GetValue()地址,而不是通过虚函数指针__vfptr去获得GetValue()的地址。
为什么Base的构造函数与虚构函数即使调用虚函数,也是调自己的函数呢?这跟构造函数与虚构函数的调用顺序有关。子类对象构造的时候,先调父类构造函数初始化父类,再调子类构造函数初始化子类。子类对象虚构的时候,恰恰相反,先调子类对象的虚构函数,再调父类的虚构函数。输出的结果也证明了这点。
所以如果父类的构造函数与虚构函数是调用子类的函数,那就非常危险了。因为父类的构造函数执行时,子类的构造函数还没有执行,说明子类还没有初始化,而这时就调用子类的方法,很容易出错,甚至崩溃。父类的虚构函数执行的时候,子类的虚构函数已经执行完毕,说明子类的资源已经被释放,而这时继续执行子类的方法,也很容易崩溃。于是,C++规范为此作了此约束。
如果真的很想在构造函数内调用子类方法进行初始化,还是显示提供一个初始化函数,让子类对象实例化完后,显示调用初始化函数。
扫码关注腾讯云开发者
领取腾讯云代金券
Copyright © 2013 - 2025 Tencent Cloud. All Rights Reserved. 腾讯云 版权所有
深圳市腾讯计算机系统有限公司 ICP备案/许可证号:粤B2-20090059 深公网安备号 44030502008569
腾讯云计算(北京)有限责任公司 京ICP证150476号 | 京ICP备11018762号 | 京公网安备号11010802020287
Copyright © 2013 - 2025 Tencent Cloud.
All Rights Reserved. 腾讯云 版权所有