1. 在使用堆内存时,需要手动delete指针以防内存泄露,但是有些例外情况(例如发生异常)会使得内存资源无法释放,此外,使用者也极有可能忘记释放堆内存,因此需要" 以对象管理资源",也就是把资源放进对象内,用析构函数来确保资源被释放,这种思想被称为"资源取得时机就是初始化时机"(Resource Acquisition is Initialization; RAII,名字并没有体现这种思想)
2. 除了手动编写一个指针类以外,C++标准库提供了auto_ptr类模板(在<memory>头文件中),使用它需注意:
1). auto_ptr<T>只能由T*显示初始化或由auto_ptr<T>对象初始化
2). auto_ptr<T>只能指向堆内存,因为它的析构函数会delete它所指向的对象,因此务必不要用指向堆栈的指针初始化auto_ptr<T>
3). auto_ptr<T>没有使用计数的概念,同一块内存只能由一个auto_ptr<T>指向,auto_ptr<T>对象的赋值操作和拷贝构造会使右操作数指向NULL.由于auto_ptr<T>的复制不同于一般意义上的复制,因此它不能作为STL容器的元素.
3. 从2中可以看出,auto_ptr的使用具有一定局限性和危险性,并且它所管理的内存不能同时被多个auto_ptr引用,TR1的"引用计数型智能指针"(reference-counting smart poiner; RCSP) shared_ptr就是一个很好的替代品."TR1代表'Technical Report 1',那是C++程序库工作小组对该份文档的称呼",TR1是C++ 03标准的一个扩展,它并不属于C++ 03标准,只是一份草稿文件,但它所提出的项目很可能成为官方标准,它叙述了14个新组件,统统放在命名空间std内,其中包括shared_ptr和weak_ptr,gcc和vs2008SP2以上已经可以支持它了gcc需要include头文件<tr1/memory>,vs需要include头文件<memory>,有的编译器需要装boost库并且include<boost/shared_ptr.hpp>和<boost/weak_ptr.hpp>才可以使用(需加命名空间限定boost::)
shared_ptr<T>的使用比较简单,它具有引用计数功能,通过使用use_count()观测资源的引用计数,使用unique()观测判断引用计数是否为1(若为1返回true),使用reset()将指针重设(默认设为空)(详细成员函数介绍见http://www.cplusplus.com/reference/memory/shared_ptr/),可以共享对象的所有权,只有最后一个指向该对象的指针被撤销时,它所指向的堆内存才会被释放,属于强引用(只要有一个强应用存在,所指对象就不能释放),因此不能解决循环引用的问题,以下就是一个循环引用的例子:
#include<iostream> #include<memory> using namespace std; class A; class B; class A{ public: shared_ptr<B> ptrb; ~A(){ cout << "A was destructed!" << endl; } }; class B{ public: shared_ptr<A> ptra; ~B(){ cout << "B was destructed!" << endl; } }; void foo(){ shared_ptr<A> s_ptra(new A()); shared_ptr<B> s_ptrb(new B()); s_ptra->ptrb = s_ptrb; s_ptrb->ptra = s_ptra; } int main(){ foo(); system("pause"); return 0; }