C++ 子类析构函数没有被调用

如果父类里执行了delete
而父类的析构函数没有加virtual声明的话
子类的析构函数是不会被调用的

希望在父类调用delete时,子类的析构函数被调用
则在父类的析构函数加上virtual

父类没有加virtual析构函数的情况下:

class Parent {
	Parent() { }
	~Parent() { }
	
	void deleteSelf() { delete this; }
};

class Child : public Parent {
	Child() { }
	~Child() { /* not working */ }
}

int main()
{
	Child *child = new Child;
	// 此处不会调用Child的析构函数
	child->deleteSelf();
}

父类加了virtual析构函数的情况下:

class Parent {
	Parent() { }
	virtual ~Parent() { }
	
	void deleteSelf() { delete this; }
};

class Child : public Parent {
	Child() { }
	~Child() { /* working */ }
}

int main()
{
	Child *child = new Child;
	// 此处会调用Child的析构函数
	child->deleteSelf();
}
### C++子类析构函数调用顺序与实现细节 在 C++ 编程语言中,当涉及继承关系时,析构函数调用顺序遵循特定规则。以下是关于子类析构函数调用顺序及其背后实现机制的详细说明。 #### 1. 析构函数调用顺序 对于具有继承层次结的对象,在其生命周期结束时,析构函数会按照 **从最派生类到基类** 的顺序依次调用[^5]。这意味着: - 如果存在多层继承,则最先调用的是最底层(即叶子节点)子类析构函数。 - 接着逐级向上调用每一层父类的析构函数,直到到达顶层基类为止。 这种逆序调用方式确保了资源释放过程中的逻辑一致性——先清理局部数据成员再处理更通用的部分。 #### 2. 虚析构函数的作用 为了保证动态分配内存并存储指针指向同类型的对象能够正确销毁整个继承链上的所有实例,通常建议将基类定义为带有 `virtual` 关键字修饰的虚析构函数[^3]。这样即使通过基类类型删除派生类对象也能触发完整的清除流程,包括调用相应的子类析构函数以及间接调用各中间层级直至根部的基础版本。 需要注意的是,如果没有显式声明成虚拟形式而仅依赖默认行为的话,在某些场景下可能导致部分重要功能缺失或者潜在错误发生[^4]。 #### 3. 自动插入父类调用 编译器内部实现了这样一个机制:每当创建一个含有公有继承特性的新类别时,它实际上会在该实体对应的破坏方法体内隐含加入对其直系祖先版块终结处理器件的一次单独呼叫语句 。例如假设有一个名为 B 的类是从另一个叫 A 的地方衍生出来的,则可能看到类似下面这样的伪代码片段表示实际生成的内容: ```cpp class B { ... public: ~B(){ // 用户自定义的操作... this->A::~A(); // 隐式的父类析构函数调用 } }; ``` 这一步骤是由标准工具完成自动化管理而开发者额外干预即可达成预期效果。 --- ### 示例程序展示具体执行路径 考虑以下简单例子来观察上述理论的实际表现情况: ```cpp #include <iostream> using namespace std; // 基础组件 class Base { protected: int baseValue; public: Base(int val):baseValue(val){ cout << "Base Constructor Called with value: "<<val<<endl; } virtual ~Base(){ cout<<"Destroying Base, Value was:"<<this->baseValue<<endl; } }; // 衍生单元之一 class DerivedOne : public Base{ private: double derivedData; public: DerivedOne(double d,int bVal):Derived(d,bVal){} explicit DerivedOne(int bVal):Base(bVal),derivedData(9876.54){} ~DerivedOne() override{ cout<<"Terminating Derived One"<<endl; } }; int main(){ DerivedOne objD1=DerivedOne(10); } ``` 运行以上脚本将会打印出如下消息序列验证我们前面提到的原则成立与否: ``` Base Constructor Called with value: 10 Terminating Derived One Destroying Base, Value was:10 ``` 从中可以看出先是完成了针对特殊化形态的数据初始化动作之后才进入常规项目建立阶段;而在最终回收环节则严格依照反向排列逐一拆解各个组成部分从而避免遗漏任何必要的善后措施。 ---
评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值