本文内容基本来自于《Effective C++》一书,为学习后的笔记,以便温故。
《Effective C++ 》第 3节 资源管理
(1)条款13:以对象管理资源
为防止资源泄漏,请使用RAII(Resource Acquisition Is Initialliztion资源取得时机便是初始化时机)对象,它们在构造函数中获得资源并在析构函数中释放资源。
两个常被使用的RAII classes分别时tr1::shared_ptr(引用计数型智慧指针RCSP)和auto_ptr(标准库中的智能指针)。前者通常是较佳选择,因为其copy行为比较直观。若选择auto_ptr,复制动作会使他指向NULL。
(2)条款14:在资源管理类中小心copying行为
复制RAII对象必须一并复制它所管理的资源,所以资源的copying行为决定RAII对象的copying行为。(例如:常见的一个错误就是,对象A中存在对另外一个对象B的引用,此时使用默认的copy函数复制该对象A为C,那么在C析构的时候会将B删除掉,A在析构的时候就会出错)
普遍而常见的RAII class copying行为是:抑制copying, 施行引用计数法(shared_ptr类似方法)。不过其他行为也可能被实现。
(3)条款15:在资源管理类中提供对原始资源的访问
APIs往往要求访问原始资源,所以每一个RAII class应该提供一个“取得其所管理之资源”的办法。(例如:shared_ptr,auto_ptr重载了指针取值操作符( -> 和 *),或者提供一个get()函数,返回一个原始指针)
对原始资源的访问可能经由显示转换或者隐式转换。一般而言显式转换比较安全,但是隐式转换对客户比较方便。
(4)条款16:成对使用new和delete时要采取相同的形式
int* p = new int[5]; delete p; //这样是错误的,只删除了第一个元素的内存空间。
如果你在new表达式中使用[],必须在相应的delete表达式中也使用[],如果你在new表达式中不使用[],也一定不要在delete中使用[]
(5)条款17:以独立语句将newed对象置入智能指针。
例如:函数 void processWidget(std::tr1::shared_ptr<Widget> pw, int priority);在具体调用的时候使用如下方式:
processWidget(std::tr1::shared_ptr<Widget> (new Widget), priority);该函数中在调用该函数之前需要做三件事情:
A :调用 priority B:执行 new Widget C: 调用tr1::shared_ptr构造函数
此时,如果这样对函数进行调用,会有资源泄漏的可能性。因为C++编译器对于这三个事情的执行顺序没有严格的规定。如果 new在第一个位置执行,priority在第二个执行,此时在执行第二个出错的时候,new出来的资源就无法放入智能指针中,也就造成了资源泄漏。
以独立的语句将newed对象存储于智能指针中,如果不这样做一旦抛出异常,就有可能导致资源泄漏。