为什么当类A的析构函数为虚函数或非虚函数时,B的类成员n具有不同的值
why the calss member n of B have different value , when destructor of class A is virtual or non virtual?
#include <iostream>
using namesapce std;
class A
{
public:
virtual ~A(){cout<<"delete A"<<endl};
};
class B: public A
{
public:
B(int n):n(n){}
void show(){cout<<n<<endl;}
~B(){cout<<"delete B"<<endl;}
private:
int n;
}
int main()
{
A *pa;
B *pb = new B(1);
pa = pb;
delete pa;
pb->show();
return 0;
}
当A类的析构函数为 virtual ~A(){...}
时,程序输出:
delete B
delete A
1
当A类的析构函数为 ~A(){...}
时,程序输出:
delete A
0
当类A的析构函数是虚的或非虚的时,为什么n的值不同?当调用B的析构函数销毁对象时,为什么类成员n仍然存在? 您的程序显示未定义行为。这是你"为什么"问题的唯一答案。
首先,如果A
的析构函数不是虚函数,那么在代码中执行delete pa
会导致未定义的行为。试图通过指向父类型A *
的指针来访问B
类型的delete
对象,导致A
的析构函数的未定义行为不是虚的。
其次,pb->show()
似乎是试图调用一个已经被销毁的对象的方法。无论析构函数是否为虚函数,其行为也是未定义的。
A *pa;
B *pb = new B(1);
pa = pb;
这被称为向上转换。无论何时进行上转换,基类析构函数都应该是虚的。
如果没有虚基类析构函数,delete pa将只调用基类析构函数,这是不希望的,因为它派生的类对象将永远不会被销毁,并导致内存泄漏。
基类的虚析构函数将首先调用派生类的析构函数,然后它将自己析构,这是期望的行为,并且不会因为上转换而导致任何泄漏。
相关文章:
- 如何使用指针传递给函数的数组中对象的函数成员
- c++构造函数成员初始化:传递参数
- 创建 std::函数,它返回具有函数成员值的变量.分段错误
- 如何在C++通过公共函数访问私有函数成员?
- 解释了构造函数成员初始化列表
- 调用std::函数成员时内存损坏
- 是否可以为模板类的模板函数成员设置别名?
- 捕获 lambda 函数C++成员变量
- 构造函数成员初始值设定项跨成员列出,安全吗?
- 获取与在模板参数中传递的函数成员类型相同的类
- 如何从公共函数成员访问地图私有成员
- C 构造函数成员分配优化
- 使用命名空间进行函数成员定义
- 函数成员作为 CUDA 内核的参数
- 模板基类函数成员的别名
- 函数成员中用于void和继承的enable_if
- 头文件中是否定义了一个很长的Class函数成员
- 类内/构造函数成员初始化
- 使用指向部分专用函数成员的指针自动填充向量
- 指向函数成员的指针