本篇文章有三个重点
1.为多态基类声明为virtual析构函数
原因很简单,在释放基类指针时会先调用继承类的析构函数。
只有当class内含至少一个virtual函数时,才能将其声明为virtual析构函数。无故声明virtual会增加内存开销。对于纯虚析构函数,在声明为纯虚析构函数后还需要为其提供一份定义,因为析构函数的运作方式是最深层派生的class其析构函数最先被调用,然后是每一个基类的析构函数被调用。
2.别让异常逃离析构函数
在析构函数的异常行为容易导致内存泄漏等不明确行为,因此建议析构函数需要对可能的异常行为进行处理
class DBConn{
public:
~DBConn()
{
try {
db.close();
}
catch(..)
{
}
};
}
原则上尽量不要讲有风险的行为放置到析构函数中,可以提供一个接口执行该操作。
class Transaction
{
public:
Transaction();
virtual void logTransaction() const = 0;
};
Transaction::Transaction()
{
...
logTransaction(); //在构造函数中调用virtual函数
}
void BuyTransaction::public Transaction
{
public:
virtual void logTransaction() const;
}
此时如果构造一个BuyTransaction对象,首先调用基类的构造函数。
基类的在构造时,此时的对象类型是基类对象而非派生类对象,因此会直接调用基类的虚函数。
对于析构函数同样如此,在进入基类的析构函数后会对象会变为基类对象。
因此不要在构造和析构函数里调用virtual函数。