大家好:
衷心希望各位点赞。
您的问题请留在评论区,我会及时回答。
虚析构和纯虚析构
多态使用时,父类指针在释放时无法调用子类的析构函数,这样的话,如果子类中有属性开辟到堆区,无法调用子类的析构函数销毁子类的堆区数据,就会造成内存泄漏。
解决方式:将父类中的析构函数改为虚析构或者纯虚析构。
虚析构和纯虚析构的共性:
可以解决父类指针释放子类对象。
都需要有具体的函数实现。
虚析构和纯虚析构的区别:
如果是纯虚析构,该类属于抽象类,无法实例化对象。
1.将父类的析构函数改为虚析构
#include <iostream>
#include <Windows.h>
#include <string>
using namespace std;
class Animal
{
public:
Animal()
{
cout << "父类构造函数调用" << endl;
}
// 虚析构
virtual ~Animal()
{
cout << "父类虚析构函数调用" << endl;
}
// 纯虚函数
virtual void speak() = 0;
};
//子类
class Cat : public Animal
{
private:
string* m_Name;
public:
Cat(string name)
{
// 子类开辟堆区数据
m_Name = new string(name);
cout << "子类的构造函数调用" << endl;
}
~Cat()
{
if (m_Name != NULL)
{
delete m_Name;
m_Name = NULL;
cout << "子类的析构函数调用" << endl;
}
}
// 重写父类的纯虚函数
virtual void speak()
{
cout << *m_Name << "小猫在说话" << endl;
}
};
void doWork(string& name)
{
Animal* animal = new Cat(name);
animal->speak();
delete animal;
animal = NULL;
}
int main(void)
{
string name = "Tom";
doWork(name);
system("pause");
return 0;
}
运行截图:
父类Animal的析构函数加上了 virtual 关键字,称为“虚析构”,这样做的好处是:可以通过父类指针释放子类的堆区数据。
在doWork函数中执行delete animal这行代码时,先执行子类Cat的析构函数~Cat()释放子类的堆区数据m_Name,然后执行父类Animal的虚析构函数~Animal()。
2.将父类的析构函数改为纯虚析构
#include <iostream>
#include <Windows.h>
#include <string>
using namespace std;
class Animal
{
public:
Animal()
{
cout << "父类构造函数调用" << endl;
}
// 纯虚析构:在类内声明,类外实现
// 有了纯虚析构的类是抽象类,不能实例化对象
virtual ~Animal() = 0;
// 纯虚函数
virtual void speak() = 0;
};
Animal::~Animal()
{
cout << "父类的纯虚析构函数调用" << endl;
}
//子类
class Cat : public Animal
{
private:
string* m_Name;
public:
Cat(string name)
{
// 子类开辟堆区数据
m_Name = new string(name);
cout << "子类的构造函数调用" << endl;
}
~Cat()
{
if (m_Name != NULL)
{
delete m_Name;
m_Name = NULL;
cout << "子类的析构函数调用" << endl;
}
}
// 重写父类的纯虚函数
virtual void speak()
{
cout << *m_Name << "小猫在说话" << endl;
}
};
void doWork(string& name)
{
Animal* animal = new Cat(name);
animal->speak();
delete animal;
animal = NULL;
}
int main(void)
{
string name = "Jerry";
doWork(name);
system("pause");
return 0;
}
运行截图:
总结
1.虚析构或纯虚析构的作用:通过父类指针释放子类对象的堆区数据。
2.如果子类中没有堆区数据,可以不写虚析构和纯虚析构。
3.拥有纯虚析构的类是抽象类,无法实例化对象。