假期归来第一天!!!!学习起来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_ ptr
和auto_ ptr
都提供一个get
成员函数用来执行显式转换,也就是它会返回智能指针内部的原始指针(的复件) :
int days = daysHeld (pInv .get()); //很好,将pInv内的原始指针传给daysHeld
就像(几乎)所有智能指针一样,tr1::shared_ ptr
和auto_ 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、对原始资源的访问可能通过显式转换或隐式转换,一般而言显示转换比较安全,但隐式转换对客户更方便。