从理论上来讲,构造函数和析构函数是可以调用虚函数的,但实际应用上不应该调用
在Effective C++09中有说明了一这点,当派生类继承了基类的时候,生成一个派生类对象的时候,首先会调用基类的构造函数生成基类的部分,但构造函数里面如果有虚函数,这时候调用的是基类的虚函数,不会下降到派生类的阶层,可以从以下几个角度理解
- 在执行基类构造函数的时候,派生类的成员变量完全未初始化,那么如果此时调用的是派生类的虚函数,如果里面调用了派生类的成员变量,那么就导致出现错误
- 派生类对象在基类构造期间,会被认为是一个基类对象,而不是派生类对象,这样设置的原因是对象的派生类部分尚未被初始化,所以面对它们,最安全的做法就是无视它们,对象在派生类构造函数执行之前不会成为一个派生类对象
- 析构函数也同样适用于上面的理由
#include<iostream>
using namespace std;
class Base
{
public:
Base()
{
Function();
}
~Base(){
Function();
}
virtual void Function()
{
cout << "Base::Fuction" << endl;
}
};
class Son : public Base
{
public:
Son()
{
Function();
}
~Son(){
Function();
}
virtual void Function()
{
cout << "A::Fuction" << endl;
}
};
int main()
{
Son s;
}
可以看到,代码能够运行,但每次调用的都是各自的虚函数,这样当你在实际运行中,生成一个派生类对象,却调用了基类的虚函数,就有可能造成错误