Effective C++之条款15:在资源管理类中提供对原始资源的访问

声明:

  1. 文中内容收集整理自《Effective C++(中文版)第三版》,版权归原书所有。
  2. 本内容在作者现有能力的基础上有所删减,另加入部分作者自己的理解,有纰漏之处敬请指正。

条款15:在资源管理类中提供对原始资源的访问

Provide access to raw resources in resource-managing classes.


使用资源管理类(resource-managing classes),可以有效的帮助我们对抗资源泄漏。 
但是,举个例子说话,在条款13中我们引入:使用智能指针auto_ptr或者tr1::shared_ptr来保存factory函数例如createInvestment的调用结果:

std::tr1::shared_ptr<Investment> pInv(createInvestment());

假设我们希望用某个函数来处理Investment对象:

int dayHeld(const Investment* pi);      //返回投资的天数

此时,如果我们想要这样去调用:

int days = daysHeld(pInv); //错误!!

这样不会通过编译,因为daysHeld需要的是Investment*的指针,而此时传递给它的类型却为tr1::shared_ptr的对象。

这时候需要一个函数,可以将RAII classes对象转换为其所内含之原始资源,本例中为tr1::shared_ptr -> Investment*。有两种方法可以实现这样的功能:显式转换和隐式转换

显式转换

tr1::shared_ptr和auto_ptr都提供了一个get成员函数,用来执行显式转换。也就是它会返回智能指针内部的原始指针(的复件):

int days = daysHeld(pInv.get()); //将pInv内的原始指针传给了daysHeld

隐式转换

就像几乎所有的智能指针一样,tr1::shared_ptr和auto_ptr也重载了指针取值(pointer dereferencing)操作符(operator ->和operator*),它们允许隐式转换至底部的原始指针:

class Investment {    //investment继承体系的根类
public:
    bool isTaxFree () const;
    ...
};

Investment* createInvestment(); //factory函数

std::tr1::shared_ptr<Investment> pi1(createInvestment()); //令tr1::shared_ptr管理一笔资源
bool taxable1 = !(pi1->isTaxFree());  //经由operator->访问资源
...

std::auto_ptr<Investment> pi2(createInvestment());  //令auto_ptr管理一笔资源
bool taxable2 = !((*pi2).isTaxFree());   //经由operator*访问资源

是否应该提供一个显式转换函数(例如get成员函数)将RAII class转换为其底部资源,还是提供隐式转换,取决于其执行的具体功能,具体说来:让接口容易被正确使用,不易被误用。因此,通常显式转换函数如get就是比较好的方法,因为:显式转换将非故意的类型转换的发生的可能性最小化了。
RAII classes并不是为了封装,而是为了确保:资源释放这一个特殊行为。 

请记住:

  1. APIs往往要求访问原始资源(raw resources),所以每一个RAII class应该提供一个“取得其所管理的资源”的办法。
  2. 对原始资源的访问可能经由显式转换或隐式转换。一般而言显式转换比较安全,但隐式转换对用户而言更为方便。
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值