天天看點

effective c++ 條款15:在資源管理類中提供對原始資源的通路

記住:

  • 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就變成虛挂着的對象了。
           

繼續閱讀