条款 14:在资源管理类中小心 copying 行为

条款 14:在资源管理类中小心 copying 行为

Think carefully about copying behavior in resource-managing classes.

class Lock
{
public:
    explict Lock(Mutex* pm) : _mutexPtr(pm) { lock(_mutexPtr); }
    ~Lock() { unlock(_mutexPtr); }
private:
    Mutex* _mutexPtr;
};

Mutex m;
{
    Lock ml(&m);
    ...
}
Lock ml1(m);
Lock mk2(ml1);

当一个 RAII 对象被复制时会发生什么事?有几种可能

禁止复制

有时候让 RAII 对象支持复制很不合理,比如 Lock 这种同步化器物(synchronize promitives),此时应该禁止复制,见条款 06

class Lock : private Uncopyable { ... };
对底层资源使用引用计数法

可以将 Mutex* 改为 shared_ptr,然而 shared_ptr 的缺省行为是引用次数为 0 时删除所指之物,而这里我们希望当引用计数为 0 时 unlock,此时我们需要用到 share_ptr 默认省略的第二参数 deleter,用于指定删除动作

class Lock
{
public:
    explict Lock(Mutex* pm) : _mutexPtr(pm, unlock) { lock(_mutexPtr.get()); }
private:
    std::shared_ptr<Mutex> _mutexPtr;
};

当 _mutexPtr 引用计数为 0 时,会调用 unlock,这里不再需要 Lock 的析构函数来 unlock

复制底部资源

当你需要资源管理类的唯一理由是希望不需要某个副本时将其释放,则可以使资源管理类再复制时做深度复制(deep copying)行为

转移底部资源的所有权

某些情况会需要确保永远只有一个 RAII 对象指向一个资源,即使 RAII 对象被复制依然如此,此时资源的拥有权会从被复制物转移到目标物,正如 auto_ptr 所实现

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值