c++ 有趣的动态转换之 delete 崩溃探究兼谈基类虚析构的重要性

简介: c++ 有趣的动态转换之 delete 崩溃探究兼谈基类虚析构的重要性

前言

《有趣的动态转换》 这篇文章中,运行 测试代码3 会崩溃。本文试图揭示崩溃的原因。

错误更正

在开始之前,需要更正《C++ 虚函数简介》中的一个错误。关于 CBaseCDerived 的虚表内容,析构函数的位置并不是直接存储了虚函数的地址,而是存储了一段编译器生成的函数,该函数内部会调用对应的析构函数。

view-CBase-CDerived-vtable-in-ida.png

所以正确的虚表应该是下面这样的:

CBase-CDerived-vtable.png

注意:debug 版默认会引入另外一层间接层,而 release 版不会。

错误回顾

回顾一下 测试代码3 运行后的错误提示,如下图:

delete-pBaseA-result.png

这是一个栈平衡被破坏的错误。在 vs 中单步调试可以知道是在执行 delete(pBaseA); 的时候导致的错误。奇怪的是,在崩溃之前,还输出了一个 NewB::PerfectFunctionName。光看源码,看不出什么问题了,需要查看反汇编代码了。

delete 的反汇编代码

disassembly-code-of-delete-pBaseA.png

根据上图中的解释,执行 delete (pBaseA); 会输出 NewB::PerfectFunctionName 已经很清楚了。但是为什么会崩溃呢?不知道有没有小伙伴儿注意到那个奇怪的 push 1。函数 NewB::PerfectFunctionName() 是没有参数的,而这里的 push 1 却向栈上压入了一个参数,所以栈就不平衡了。

至此,执行 delete (pBaseA); 会输出 NewB::PerfectFunctionName 并且崩溃的来龙去脉应该已经清楚了。但是那个 push 1 到底是什么呢?

奇怪的 push 1

为了弄清这个 push 1 的来历与作用,我把 delete pBaseA 改成了 delete((BaseB*)pBaseA);,这样代码会按正常的逻辑执行。 也就是会执行到 NewB::'vector deleting destructor'。查看对应的反汇编代码,如下图:

NewB-vector-deleting-destructor.png

从图中高亮的三句反汇编语句可知:NewB::vector deleting destructor 需要一个参数。该参数是一个标记,如果为 1,则调用 operator delete 释放内存,否则不释放内存。

从整个反汇编代码可知,NewB::vector deleting destructor 会先执行 NewB::~NewB(),然后根据外部传入的标记来决定是否调用 operator delete 释放内存。

至此,理清了 push 1 的用途,那什么时候会 push 0 呢?

不知道有没有小伙伴儿显式调用过析构函数,像下面这样。

manually-call-destructor.png

如果查看 pBaseB->~BaseB() 的反汇编代码,一切都会真相大白。如下图:

disassembly-code-of-manually-call-destructor-and-delete.png

为什么多态基类的析构函数要是虚的?

相信有经验的 C++ 开发人员一定听过类似的忠告:带有多态性质的基类应该声明一个虚析构函数。如果类带有任何虚函数,它就该拥有一个虚析构函数。

如果析构函数不是虚函数呢?会有什么问题吗?稍微改动一下测试代码,如下:
delete-non-virtual-destructor-base-class-pointer.png

运行结果如下图:

delete-non-virtual-destructor-base-class-pointer-result.png

只有基类的析构函数被调用,子类的析构函数并没有被调用!为什么会这样呢?真相就在反汇编代码里:

disassembly-code-of-delete-non-virtual-destructor-base-class-pointer

从上图可知,如果要 delete 的类型的析构函数是非虚的,那么 vs 中带的编译器在生成汇编代码时,会直接调用对应类型的 scalar deleting destructor,不存在多态行为!这会导致子类的析构函数没有被调用!

总结

  • 如果一个类会被当成基类使用,请确保其析构函数是虚函数。

  • 在生成 delete (pBaseA); 这条语句的汇编代码时,编译器是根据 pBaseA 的静态类型确定虚析构函数在虚表中的位置的。而不是根据 pBaseA 实际指向的类型。

  • delete pBaseA 会先执行 pBaseA 指向的类型的析构函数,然后再调用 operator delete 释放对应的内存。
  • 可以显式调用一个类的析构函数。当然,析构函数的访问级别必须是 public 的。

参考资料

  • vs 反汇编代码

  • 《effective c++》

相关文章
|
17小时前
|
编译器 程序员 C语言
从C语言到C++⑨(第三章_C&C++内存管理)详解new和delete+面试题笔试题(下)
从C语言到C++⑨(第三章_C&C++内存管理)详解new和delete+面试题笔试题
5 0
|
17小时前
|
编译器 C语言 C++
从C语言到C++⑨(第三章_C&C++内存管理)详解new和delete+面试题笔试题(中)
从C语言到C++⑨(第三章_C&C++内存管理)详解new和delete+面试题笔试题
4 0
|
1天前
|
安全 程序员 编译器
C++程序中的基类与派生类转换
C++程序中的基类与派生类转换
8 1
|
1天前
|
C++ 开发者
C++程序中利用虚函数实现动态多态性
C++程序中利用虚函数实现动态多态性
9 2
|
6天前
|
存储 安全 Java
C++ delete语句
C++ delete语句
6 0
|
6天前
|
存储 编译器 C++
【C++】内存管理和模板基础(new、delete、类及函数模板)
【C++】内存管理和模板基础(new、delete、类及函数模板)
25 1
|
6天前
|
安全 程序员 C++
C++ new和delete的用法
需要注意的是,使用 `new`和 `delete`分配和释放内存时,程序员负责管理内存的分配和释放,这可能导致内存泄漏或释放已释放内存的问题。因此,C++引入了智能指针(如 `std::shared_ptr`和 `std::unique_ptr`)以更安全和自动化地管理内存。
36 2
|
6天前
|
消息中间件 算法 Java
C++实时通信优化技术探究
C++实时通信优化技术探究
25 3
|
6天前
|
C++
C++中的 虚析构 与 纯虚析构
C++中的 虚析构 与 纯虚析构
|
6天前
|
C++
3. C++构造和析构
3. C++构造和析构
31 0