构造函数和析构函数中能否调用虚函数?()

在C++ primer中说到过是最好不要调用,不是不能调用,所以构造函数跟虚构函数里面都是可以调用虚函数的,并且编译器不会报错。但是在基类中声明纯虚函数并且在基类的析构函数中调用,编译器会报错。

但是由于类的构造顺序先构造基类然后再派生类,所以在构造函数中调用虚函数,虚函数是不会呈现出多态的

类的析构顺序是先析构派生类然后再析构基类,所以当调用继承层次中某一层次的类的析构函数时,这代表其派生类已经进行了析构,所以也并不会呈现多态

案例如下:

class A
{
public:
    A()
    {
        Fuction();
    }
    virtual void Fuction()
    {
        cout << "A::Fuction" << endl;
    }
};
 
class B : public A
{
public:
    B()
    {
        Fuction();
    }
 
    virtual void Fuction()
    {
        cout << "B::Fuction" << endl;
    }
};

B b;

输出结果是:A::Fuction B::Fuction

为什么呢?

当在构造基类部分时,派生类还没被完全创建。即当A::A()执行时,B类对象还没被完全创建,此时它被当成一个A对象,而不是B对象,因此Function()绑定的是A的Function()。

基类部分在派生类部分之前被构造,当基类执行构造函数时,派生类中的数据成员还未被初始化。如果在基类构造函数中调用虚函数被解析成调用派生类的虚函数,而派生类的虚函数中又访问到未初始化的派生类数据,这是危险的,将会导致程序出现未知行为及bug。

————————

在C++ primer中说到过是最好不要调用,不是不能调用,所以构造函数跟虚构函数里面都是可以调用虚函数的,并且编译器不会报错。但是在基类中声明纯虚函数并且在基类的析构函数中调用,编译器会报错。

但是由于类的构造顺序先构造基类然后再派生类,所以在构造函数中调用虚函数,虚函数是不会呈现出多态的

类的析构顺序是先析构派生类然后再析构基类,所以当调用继承层次中某一层次的类的析构函数时,这代表其派生类已经进行了析构,所以也并不会呈现多态

案例如下:

class A
{
public:
    A()
    {
        Fuction();
    }
    virtual void Fuction()
    {
        cout << "A::Fuction" << endl;
    }
};
 
class B : public A
{
public:
    B()
    {
        Fuction();
    }
 
    virtual void Fuction()
    {
        cout << "B::Fuction" << endl;
    }
};

B b;

输出结果是:A::Fuction B::Fuction

为什么呢?

当在构造基类部分时,派生类还没被完全创建。即当A::A()执行时,B类对象还没被完全创建,此时它被当成一个A对象,而不是B对象,因此Function()绑定的是A的Function()。

基类部分在派生类部分之前被构造,当基类执行构造函数时,派生类中的数据成员还未被初始化。如果在基类构造函数中调用虚函数被解析成调用派生类的虚函数,而派生类的虚函数中又访问到未初始化的派生类数据,这是危险的,将会导致程序出现未知行为及bug。