Effective C++ 条款13,14

条款13:以对象管理资源

void f(){
	Investment* pInv=createInvestment();//调用factory函数
	...
	delete pInv;
}

该段代码可能存在的问题:

  • "…"区域内有一个过早的return语句
  • “…”区域内的语句抛出异常

将会使得资源泄露。为确保资源总是被释放,可把资源放进对象内 ,依赖C++的“析构函数自动调用机制”确保资源被释放。

智能指针的实现便应用了以上思想:

void f(){
	std::auto_ptr<Investment>pInv(createInvestment());
	...//经由auto_ptr的析构函数自动删除pInv
}

以“对象管理资源”的思想有两个关键想法:

  • 获得资源后立刻放进管理对象内(RAII)。 即用资源初始化或赋值对象
  • 管理对象运用析构函数确保资源被释放。

两种智能指针(auto_ptr与shared_ptr)的区别:

std::auto_ptr: 若通过copy构造函数或copy assignment操作符复制它们,它们会变成null,而复制所得的指针将取得资源的唯一拥有权。

std::auto_ptr<Investment>
	pInv1(createInvestment());//pInv1指向createInvestment返回物
	std::auto_ptr<Investment>pInv2(pInv1);//现在pInv2指向对象,pInv1被设为nullptr
	pInv1=pInv2;//现在pInv1指向对象,pInv2被设为nullptr

std::shared_ptr: “引用计数型智慧指针”,持续追踪共有多少对象指向某笔资源,并在无人指向它们时自动删除该资源。

std::shared_ptr<Investment>pInv1(createInvestment());//pInv1指向createInvestment()返回对象
std::shared_ptr<Investment>pInv2(pInv1);//pInv1和pInv2指向同一个对象
pInv1=pInv2;//同上

注意:
auto_ptr和shared_ptr两者都在其析构函数内做delete而不是delete[]动作。因此在动态分配而得的array身上使用auto_ptr或shared_ptr是馊主意!

请注意:

  • 为防止资源泄露,请使用RAII对象,它们在构造函数中获得资源并在析构函数中释放资源。
  • 两个常被使用的RAII classes分别是auto_ptr和shared_ptr,后者通常是较佳选择,因为其copy行为比较直观。若选择auto_ptr,复制动作会使它(被复制物)指向null。

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

当一个RAII对象被复制时,你会怎么做?

  • 禁止复制。 将copying行为设为private,具体做法参考条款6。
  • 对底层资源祭出“引用计数法”。 可利用shared_ptr实现。但需要注意,某些情况下使用RAII对象并不是要将资源释放,而是要确保调用资源的某些功能,此时需要指定一个“删除器(deleter)”——一个函数或函数对象作为参数给shared_ptr。
  • 复制底部资源。 如果允许复制资源管理对象,则你需要进行“深拷贝”。
  • 转移底部资源的拥有权。 auto_ptr的思想。

请记住:

  • 复制RAII对象必须一并复制它所管理的资源,所以资源的copying行为决定RAII对象的copying行为。
  • 普遍而常见的RAII class copying行为是:抑制copying、施行引用计数法。不过其他行为也都可能被实现。
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值