Effective.C 读书笔记04

条款07:为多态基类声明virtual 析构函数

class TimeKeeper
{
public :
	TimeKeeper();
	virtual ~TimeKeeper();
	...
};
class AtomicClock:public TimeKeeper{ ... };
TimeKeeper *ptk = getTimeKeeper();
delete ptk;
为base 声明virtual析构函数。如果class 带有任何virtual 函数,它就应该拥有一个virtual 析构函数。这样才能正确的释放derived classes 的内存。

当正常的classes 不需要声明virtual 析构函数,不然会增加内存开销。

心得:只有当class内含至少一个virtual 函数,才为它声明virtual 析构函数。

请记住:带多态性质的base classes 应该声明一个virtual 析构函数。如果class 带有任何virtual 函数,它就应该拥有一个virtual 析构函数。

               classes 的设计目的如果不是作为 base classes 使用,或不是为了具备多态性,就不该声明virtual 析构函数。

条款08:别让异常逃离析构函数

假设有个class负责数据库连接

class DBConnection
{
	public DBConnection create();
	void close();
};

class DBConn   //这个class用来管理DBConnection对象
{
public :
	...
	~DBConn()          //确保数据库被关闭
	{
		db.close();
	}
private:
	DBConnection db;
}
//这样会在析构函数抛出异常
DBConn::~DBConn()
{
	try{db.close();}
	catch()
	{
		//制作运转记录,记下对close的调用失败
		std::abort();
	}
}
//这样会导致不明确的行为
DBConn::~DBConn()
{
	try{db.close();}
	catch()
	{
		//制作运转记录,记下对close的调用失败
	}
}
//将异常吞下也不好

稍微好点的做法是:

class DBConn
{
public:
	...
	void close()           //供客户使用的新函数
	{
		db.close();
		closed = true;
	}
	~DBConn()
	{
		if(!closed)
		{
			try                 //关闭连接(如果客户不那么做的话)
			{
			    db.close();
			}
			catch (...)          //如果关闭动作失败
			{
				制作运转记录,记下对close 的调用失败//记录下来并结束程序或吞下异常
                ...
			}


		}
	}
private:
	DBConnection db;
	bool closed;

};

把调用close的责任从DBConn析构函数手上移到DBConn客户手上。

请记住:析构函数绝对不要吐出异常,如果一个被析构函数调用的函数可能抛出异常,析构函数应该捕捉任何异常,然后吞下他们(不传播)或结束程序。

                如果客户需要对某个操作函数运行期间抛出的异常做出反应,那么class 应该提供一个普通函数(而非在析构函数中)执行该操作。


你可能感兴趣的:(Effective.C 读书笔记04)