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

Effective C++ 详解条款15: 在资源管理类中提供对原始资源的访问
假期归来第一天!!!!学习起来up up up ღ( ´・ᴗ・` )ღ比心

由条款13导入一个例子:使用智能指针如auto_ptr
tr1: :shared ptr保存factory函数如createInvestment的调用结果:

std::tr1::shared_ ptr<Investment> pInv (createInvestment()); //见条款13

假设你希望以某个函数处理Investment对象,像这样:

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

你想要这么调用它:

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

却通不过编译,因为daysHeld需要的是Investment*指针,你传给它的却是个类型为tr1::shared_ ptr<Investment> 的对象。

做法:显式转换和隐式转换

目的需要一个函数可将RAII class对象(本例为tr1::shared_ ptr)转换为其所内含之原始资源(本例为底部之Investment*)。

方法一:显示转换,tr1::shared_ ptrauto_ ptr都提供一个get成员函数用来执行显式转换,也就是它会返回智能指针内部的原始指针(的复件) :

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

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

class Investment {                                                //investment继承体系的根类
public:
   bool isTaxFree() const;
   ...
};
Investment* createInvestment() ;                                 //factory函数
std::tr1::shared_ ptr<Investment> pi1 (createInvestment( )) ;    //令trl::shared_ ptr 管理-笔资源。        
bool taxablel = ! (pi1->isTaxFree()) ;                           //经由operator->访问资源。
...
std::auto_ ptr<Investment> pi2 (createInvestment());            //令 auto_ ptr管理一笔资源。
bool taxable2 = ! ((*pi2).isTaxFree());                         //经由operator*访问资源。
...

方法二:隐式转换,由于有时候还是必须取得RAII 对象内的原始资源,做法是提供一个隐式转换函数。

字体的示例,以下的显示转换增加了泄漏字体的可能性。

class Font {
public:
    ...
	FontHandle get() const { return f; }                       //显式转换函数
};
class Font {
public:
    ...
	operator FontHandle() const                               //隐式转换函数
	{ return f; }
    ...
};

总结

1、是否该提供一一个显式转换函数(例如get成员函数)将RAII class转换为其底部资源,或是应该提供隐式转换,答案主要取决于RAII class被设计执行的特定工作,以及它被使用的情况。最佳设计很可能是坚持条款18的忠告:“让接口容易被正确使用,不易被误用”。通常显式转换函数如get是比较受欢迎的路子,因为它将“非故意之类型转换”的可能性最小化了。然而有时候,隐式类型转换会增加错误发生机会。

2、APIs往往要求访问原始资源,所以每一个RAII class应该提供一个“取得他所管理的资源”的方法。

3、对原始资源的访问可能通过显式转换或隐式转换,一般而言显示转换比较安全,但隐式转换对客户更方便。

上一篇:MySQL的MDL锁


下一篇:Effective C++ 详解条款14: 在资源管理类中小心coping行为