/*
* 虚析构和纯虚析构
* 多态使用时,如果子类中有属性开辟到堆区,那么父类指针在释放时无法调用子类的析构代码
* 解决方式:将父类中的析构函数改为虚析构或者纯虚析构
* 虚析构和纯虚析构共性:
* 1、可以解决父类指针释放子类对象
* 2、都需要具体的函数实现
* 虚析构和纯虚析构区别:
* 1、如果时纯虚析构,该类属于抽象类,无法实例化对象
* 虚析构语法:
* virtual ~类名(){};
* 类名::~类名(){};//函数具体实现
* 纯虚析构语法:
* virtual ~类名(){} = 0;
* 类名::~类名(){};//函数具体实现
*/
#include<iostream>
#include<string>
using namespace std;
class Animal
{
public:
Animal()
{
cout << "Animal构造函数调用!" << endl;
}
virtual void Speak() = 0;
//析构函数加上virtual关键字,变成虚析构函数
// virtual ~Animal()
// {
// cout << "Animal虚析构函数!" << endl;
// }
//利用虚析构函数来解决通过父类指针释放子类对象不干净问题
//需要声明 也需要实现;虚函数可以只有函数声明
virtual ~Animal() = 0;//如果不是虚析构函数,无法释放子类堆区属性
};
Animal::~Animal()//函数实现
{
cout << "Animal 纯虚析构函数调用!" << endl;
}
//和包含普通纯虚函数的类一样,包含了纯虚析构函数的类也是抽象类,不能狗实例化
class Cat : public Animal
{
public:
Cat(string name)
{
cout << "Cat构造函数调用!" << endl;
m_Name = new string(name);//子类属性在堆区
}
virtual void Speak()
{
cout << *m_Name << "小猫在说话!" << endl;
}
~Cat()
{
if (this->m_Name!=NULL)
{
cout << "Cat析构函数调用!" << endl;
delete m_Name;
m_Name = NULL;
}
}
public:
string *m_Name;
};
void test04()
{
Animal* animal = new Cat("Tom");
animal->Speak();
//父类指针在析构时候,不会调用子类中析构函数,导致子类如果有堆区属性,出现内存泄露
//通过父类指针去释放,会导致子类对象可能清理不干净,造成内存泄露
//解决方法:给基类增加一个虚析构函数
//虚析构函数就是用来解决通过父类指针释放子类对象
delete animal;
}
int main()
{
test04();
cout<<"Hello World!"<<endl;
system("pause");
return 0;
}
C++学习第九十四篇
最新推荐文章于 2024-07-14 13:39:50 发布