Effective C++(14) 在资源管理类中小心copying行为

问题聚焦:
    上一条款所告诉我们的智能指针,只适合与在堆中的资源,而并非所有资源都是在堆中的。
    这时候,我们可能需要建立自己的资源管理类,那么建立自己的资源管理类时,需要注意什么呢?。

在详述这一章的主题之前,先回忆一下上一节所提到的一个名词——RAII(Resource Acquisition Is Initialization)
含义就是:资源取得时机便是初始化时机。
如果上一节对这个观念的理解还不是很深的话,那么下面这个例子可以让你更好地理解。

Demo 假设我们使用C API函数处理类型为Mutex的互斥器对象,共有lock和unlock两函数可用。
void lock(Mutex* pm);
void unlock(Mutex* pm);

为了确保不会忘记将一个被锁住的Mutex解锁,你可能会希望简历一个class用来管理锁。
这样的class的基本结构由RAII守则支配,也就是在“资源构造期间获得,在析构期间释放”
class Lock {
public:
    explicit Lock(Mutex* pm) : mutexPtr(pm)
    { lock(mutexPtr);    }     // 获得资源
    ~Lock() 
    { unlock(mutexPtr); }    // 释放资源
};

// 客户对Lock的正确用法符合RAII方式
Mutex m;         // 定义你需要的互斥器
....
{                         // 建立一个区块用来定义critical section
    Lock ml(&m);      // 锁定互斥器
    ......
}                        // 在区块最末尾,自动接触互斥器锁定

上面的用法自然没有什么问题,那么问题是什么呢?——如果Lock对象被复制,会发生什么事呢?就像下面这样:
Lock ml1(&m);
Lock ml2(ml1);


一般化这个问题就是:当一个RAII对象被复制时, 会发生什么事情呢?

大多数情况下,有两种解决方法:
1 禁止复制:复制动作对RAII class并不合理。如果阻止复制操作,可以转到:Effective C++(6) 如何拒绝编译器的自动生成函数
2 对底层资源祭出“引用计数法:
    通常,只要内含一个tr::shared_ptr成员变量,就可以实现引用技术。代码是下面这个样子的:
    class Lock {
    public:
        explicit Lock(Mutex* pm) : mutexPtr(pm, unlock)  
        {
            lock(mutexPtr.get());
        }
    private:
        std::tr1LLshared_ptr<Mutex> mutexPtr;
    };

需要注意的一点是:tr1::shared_ptr的缺省行为是“当引用次数为0时,删除其所指对象”,或许这并不是我们想要的行为。幸运的是,tr1::shared_ptr并不是只能删除对象,而是允许指定我们想要的动作,只需要在第二个参数上传递一个对象或函数对象,当引用次数为0时,便被调用。
在本例中,传递的就是解锁函数,而不是删除所指对象。

遇到RAII的复制动作时,我们还可以有别的处理方式:
1 复制底部资源
    复制资源管理对象时,同时也可以复制其所包含的资源,因此进行的应该是“深拷贝”。
    例如:当一个对象包含一个指针指向一块堆内存时,复制这个对象,同时其指针和指向的内存都会被复制出一个复件。
2 转移底部资源的拥有权
    在某些场合下,你可能希望确保永远只有一个RAII对象指向一个未加工的资源,就像auto_ptr的复制行为。详见: Effective C++(13) 用对象管理资源

小结:
  • 复制RAII对象必须一并复制它所管理的资源,所以资源的拷贝行为决定RAII对象的拷贝行为
  • 一般的复制行为是:阻止拷贝行为,使用引用计数法(tr1::shared_ptr)

参考资料:
《Effective C++ 3rd》


Effective C++(14) 在资源管理类中小心copying行为

上一篇:C++ 算法库(2) 修改内容的序列操作


下一篇:linux下安装ntop