想通过父类指针 把所有的子类对象的析构函数都执行一遍释放所有的子类资源,就需要使用虚析构函数
#define _CRT_SECURE_NO_WARNINGS
#include <iostream>
using namespace std;
//虚析构函数
class A
{
public:
A()
{
p = new char[20];
strcpy(p, "obja");
printf("A()\n");
}
virtual ~A()//在父类的析构函数之前加上virtual构成虚析构函数
{
delete[] p;
printf("~A()\n");
}
private:
char *p;
};
class B : public A
{
public:
B()
{
p = new char[20];
strcpy(p, "objb");
printf("B()\n");
}
~B()
{
delete[] p;
printf("~B()\n");
}
private:
char *p;
};
class C : public B
{
public:
C()
{
p = new char[20];
strcpy(p, "objc");
printf("C()\n");
}
~C()
{
delete[] p;
printf("~C()\n");
}
private:
char *p;
};
//想通过父类指针 把所有的子类对象的析构函数都执行一遍
//想通过父类指针 释放所有的子类资源 ,就需要使用虚析构函数
void howtodelete(A *base)
{
delete base; //函数之间虽然有继承关系,但是并未构成多态,所以当子类对象传递给父类时,释放的是父类对象的内存,而不是子类对象内存。这样将会造成内存泄漏
}
void main05()
{
C *myC = new C; //利用new给C类型的指针对象分配内存空间
howtodelete(myC);//1.将子类对象传递给函数释放,若函数类之间并没有表现多态属性,那么就相当于用子类对象传递给父类对象,再释放父类对象的内存,并不会对子类产生任何作用,这样将会造成内存泄漏
//2.若在父类对象的析构函数前加上virtual,就构成了虚构函数,执行父类虚构时就会调用所有子类的析构函数。
//delete myC; //直接通过子类对象释放所有资源 不需要写virtual 。但是直接释放子类对象将在实际开发中并不会有太多意义
}