C++学习第九十四篇

/*
* 虚析构和纯虚析构
* 多态使用时,如果子类中有属性开辟到堆区,那么父类指针在释放时无法调用子类的析构代码
* 解决方式:将父类中的析构函数改为虚析构或者纯虚析构
* 虚析构和纯虚析构共性:
* 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;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值