RAII对象只能管理堆上new出来的资源,其它的不行。比如下面的mutex:
void lock(Mutex *pm); // lock mutex pointed to by pm void unlock(Mutex *pm); // unlock the mutex
只能自定义一个资源管理类,来管理这个Mutex资源,以防止它被意外地忽略。
class Lock { public: explicit Lock(Mutex *pm) : mutexPtr(pm) { lock(mutexPtr);} // acquire resource ~Lock() { unlock(mutexPtr);} // release resource private: Mutex *mutexPtr; };
使用方法如下:
Mutex m; // define the mutex you need to use ... { // create block to define critical section Lock ml(&m); // lock the mutex ... // perform critical section operations } // automatically unlock mutex at end of block
如果拷贝一个RAII资源,会发生什么?
Lock ml1(&m); Lock ml2(ml1);
一般的处理方式:
1> 因为有的拷贝RAII资源无意义,比如Lock。所以禁止拷贝:
class Lock: private Uncopyable { // prohibit copying — see Item 6 ... };
2> 如果拷贝有意义,那么用“引用计数”机制。使用tr1::shared_ptr即可:
class Lock { public: explicit Lock(Mutex *pm) // init shared_ptr with the Mutex : mutexPtr(pm, unlock) // to point to and the unlock func { // as the deleter lock(mutexPtr.get()); } private: std::tr1::shared_ptr<Mutex> mutexPtr; };
在mutexPtr的ctor里指定unlock作为计数器归零时要做的“回收资源”操作。
Lock不需要再定义dtor。编译器会自动生成。
3> 有时要求只能有一个管理器指向资源,那么可在拷贝时转移所有权。auto_ptr就是这么做的。