偷偷拿来记录一下萌新的cs路——day 42
用多态编写以下代码:
#include<iostream>
#include<string>
using namespace std;
class Animal {
public:
Animal() {
cout<<"Animal 构造函数调用"<<endl;
}
virtual void speak() = 0;
~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 << ": Miao " << endl;
}
~Cat() {
if (m_Name != NULL) {
cout << "Cat 析构函数调用" << endl;
delete m_Name;
m_Name = NULL;
}
}
string * m_Name;
};
void test01() {
Animal* animal = new Cat("Tom");
animal->speak();
delete animal;
}
int main() {
test01();
return 0;
}
运行结果:
由于Cat 析构函数未被调用,可知堆区数据未被及时删除,存在泄露。其原因是父类指针在析构时不会调用子类的析构函数,所以子类若有堆区数据会内存泄露。
解决方法:将父类析构函数改为虚函数
virtual ~Animal() {
cout << "Animal 析构函数调用" << endl;
}
运行结果:
若改为纯虚析构函数,会报错“无法解析的外部命令”,原因为父类也有需要释放的堆区数据,而析构函数未定义。
virtual ~Animal() = 0;
解决方法:在父类外定义纯虚析构函数。
class Animal {
public:
Animal() {
cout<<"Animal 构造函数调用"<<endl;
}
virtual void speak() = 0;
// 虚析构解决父类指针释放子类对象不干净问题
/*virtual ~Animal() {
cout << "Animal 析构函数调用" << endl;
}*/
virtual ~Animal() = 0;
};
Animal:: ~Animal() {
cout << "Animal 纯虚析构函数调用" << endl;
}
运行结果:
注意纯虚析构函数存在时,类同样变为抽象类,无法被实例化。
若子类不存在需要释放的堆区数据,可以不使用虚析构or纯虚析构函数。
有误之处请大佬指正,感谢!