构造函数和析构函数内部调用虚函数


在构造函数内部调用虚函数

在普通的成员函数中调用虚函数时仍然是在运行时解析虚函数调用,因为此时对象并不知道其是属于成员函数所在的类或者是派生类。但是在构造函数中,虚机制没有作用,只有“本地”的虚函数会被调用,有两个原因:

首先,在构造函数中你只能知道基类对象已经初始化,但不知道谁将继承你,若调用派生类中对应的虚成员函数,则其可能操作尚未初始化的成员;

其次,构造函数首先初始化VPTR,此表只能根据基类和当前类来初始化,其在对象的生命周期里将不变,除非还有其他类以此为基类。如果紧接着调用下一步的构造函数,则其将VPTR指向自己的VTABLE,如此向后,直到最后一层。当进行虚函数调用时,只能利用当前的VPTR去获得对应的虚函数。


析构函数中调用虚函数

和构造函数一样,析构函数中调用虚函数,只能调用本地版本,基本原理通构造函数一样,唯一的区别是对于构造函数,类型信息尚不完全,调用析构函数时知道类型信息,但对应的虚函数不再可用,已经析构了

//: C15:VirtualsInDestructors.cpp

// Virtual calls inside destructors

#include <iostream>

using namespace std;

class Base {

public:

virtual ~Base() {

cout << "Base1()/n";

f();

}

virtual void f() { cout << "Base::f()/n"; }

};

class Derived : public Base {

public:

~Derived() { cout << "~Derived()/n"; }

void f() { cout << "Derived::f()/n"; }

};

int main() {

Base* bp = new Derived; // Upcast

delete bp;

} ///:~


运行结果:

~Derived()/n

Base1()/n

Base::f()/n





  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在 C++ ,当创建对象时,首先会调用该类的构造函数来初始化对象成员变量和数据成员。而当对象被销毁时,会自动调用析构函数来清理对象并释放相关资源。 构造函数析构函数调用顺序与对象的创建和销毁顺序密切相关。对于单个对象,构造函数调用先于析构函数。而对于多个对象,它们的构造函数析构函数调用顺序则取决于它们的创建和销毁顺序。 具体来说,创建对象时,构造函数调用顺序按照成员变量的声明顺序进行,即先调用基类构造函数,再调用成员变量的构造函数,最后调用自身的构造函数。而销毁对象时,析构函数调用顺序则与构造函数相反,即先调用自身的析构函数,再调用成员变量的析构函数,最后调用基类的析构函数。 需要注意的是,如果一个类是另一个类的成员变量,则其构造函数析构函数调用顺序取决于它们在类的声明顺序。如果一个类是另一个类的基类,则其构造函数析构函数调用顺序与继承方式有关。如果是虚继承,则先调用最远的祖先类的构造函数,再依次调用间类和自己的构造函数;而析构函数调用顺序则相反,先调用自己的析构函数,再依次调用间类和最远的祖先类的析构函数。 总之,构造函数析构函数调用顺序是 C++ 一个非常重要的概念,需要开发者在编写程序时仔细考虑。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值