C++学习笔记之虚析构和纯虚析构/虚析构和纯虚析构的学习/虚析构和纯虚析构学习
多态使用时,如果子类中有属性开辟到堆区,那么父类指针在释放时无法调用到子类的析构代码。
解决办法: 将父类中的析构函数改为虚析构或者纯虚析构
虚析构和纯虚析构共性:
可以解决父类指针释放子类对象
都需要有具体的函数实现
虚析构和纯虚析构区别:
如果是纯虚析构,该类是抽象类,无法实例化对象。
虚析构语法
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;
//}
/*~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()
{
cout << "Cat的析构函数!" << endl;
if (this->m_Name != NULL)
{
delete m_Name;
m_Name = NULL;
}
}
public:
string* m_Name;
};
void test()
{
Animal* animal = new Cat("小花");
animal->speak();
//通过父类指针去释放,会导致子类对象可能清理不干净,造成内存泄漏
//怎么解决? 给基类添加一个虚析构函数
//虚析构函数就额是用来解决通过父类指针是释放子类对象
delete animal;
}
int main()
{
test();
return 0;
}
//总结:虚析构或纯虚析构就是用来解决通过父类指针释放子类对象
//如果子类中没有堆区数据,可以不写虚析构或纯虚虚构
//拥有纯虚析构的类也属于抽象类
C++学习笔记之虚析构和纯虚析构/虚析构和纯虚析构的学习/虚析构和纯虚析构学习