记住:
- APIs往往要求访问原始资源,所以每一个RAII类应该提供一个“取得其所管理之资源”的办法。
- 对原始资源的访问可能经由显示转换或隐式转换。一般而言显示转换比较安全,但隐式转换对客户比较方便。
std::tr1::shared_ptr<Investment> pInv(createInvestment());
int daysHeld(const Investment* pi);
int days = daysHeld(pInv); //错误
int days = daysHeld(pInv.get()); //正确,shared_ptr和auto_ptr都提供一个get成员函数,返回内部的原始指针。
tr1::shared_ptr和auto_ptr重载了指针取值操作符 (operator->和operator*),它们允许隐式转换至内部原始指针。
class Investment
{
public:
bool isTaxFree() const;
...
};
Investment* createInvestment();
std::tr1::shared_ptr<Investment> pi1(createInvestment());
bool taxable1 = !(pi1->isTaxFree()); //经由operator->访问资源
...
std::auto_ptr<Investment> pi2(createInvestment());
bool taxable2 = !((*pi2).isTaxFree()); //经由operator*访问资源
...
FontHandle getFont();
void releaseFont(FontHandle fh);
class Font {
public:
explicit Font(FontHandle fh)
: f(fh)
{ }
~Font() { releaseFont(f); }
FontHandle get() const { return f; } //显示转换函数
operator FontHandle() const //隐式转换函数
{ return f; }
private:
FontHandle f;
}
//使用显示转换函数
void changeFontSize(FontHandle f, int newSize);
Font f(getFont());
int newFontSize;
...
changeFontSize(f.get(), newFontSize);
//使用隐式转换函数
Font f(getFont());
int newFontSize
...
changeFontSize(f, newFontSize);
//使用隐式转换函数,会增加错误发生的机会
Font f1(getFont());
...
FontHandle f2 = f1; //原意时要拷贝一个Font对象,却将f1隐式转换为FontHandle,然后复制它。
//如果f1被销毁,字体被释放,而f2就变成虚挂着的对象了。