このブログ投稿とこのシリーズは、「Effective C ++」を見ることについての私の考えであり、毎日C ++を改善するためのちょっとしたコツを学びます。
アイテム07:多形基本クラスの仮想デストラクタを宣言する
ポリモーフィズムを使用する場合、ヒープ領域に開かれているプロパティがサブクラスにあると、サブクラスのデストラクタコードが解放されたときに親クラスのポインタを呼び出すことができず、キーワードvirtualが親の仮想関数に追加されます。クラス。
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()
{
cout << "Cat析构函数调用!" << endl;
if (this->m_Name != NULL) {
delete m_Name;
m_Name = NULL;
}
}
public:
string* m_Name;
};
class Dog : public Animal {
public:
Dog(string name)
{
cout << "Dog构造函数调用!" << endl;
m_Name = new string(name);
}
virtual void Speak()
{
cout << *m_Name << "小狗在说话!" << endl;
}
~Dog()
{
cout << "Dog析构函数调用!" << endl;
if (this->m_Name != NULL) {
delete m_Name;
m_Name = NULL;
}
}
public:
string* m_Name;
};
void test01()
{
Animal* animal1 = new Cat("Tom");
animal1->Speak();//调用cat的speak函数
//通过父类指针去释放,会导致子类对象可能清理不干净,造成内存泄漏
//怎么解决?给基类增加一个虚析构函数
//虚析构函数就是用来解决通过父类指针释放子类对象
delete animal1;
cout << endl;
Animal* animal2 = new Dog("Ben");
animal2->Speak();//调用dog的speak函数
delete animal2;
}
実行時の影響
- C ++の仮想関数は、ランタイムポリモーフィズムを生成し、遅延バインディングの効果を実現するため、多くのC ++デザインパターンが仮想の周りにあります。
- 仮想関数として宣言されたものはすべて、32ビットシステムで4ビットを占める仮想関数テーブルポインタを伝送します。
覚えておいてください:
ポリモーフィズムを持つ基本クラスは、仮想(仮想)デストラクタを宣言する必要があります。クラスに仮想関数がある場合は、仮想デストラクタが必要です。
クラスの設計目的は、基本クラスとして使用したり、ポリモーフィズムを持たせたりすることではないため、仮想デストラクタを宣言しないでください。