【Effective C++笔记】资源管理

C++程序中最常使用的资源就是动态分配内存(如果你分配内存却从来不曾归还它,会导致内存泄露),但内存只是你必须管理的众多资源之一。其他常见的资源还包括文件描述符、互斥锁、图形界面中的字型和笔刷、数据库连接、以及网络sockets。不论哪一种资源,重要的是,当你不再使用它时,必须将它归还给系统。
尝试在任何运用情况下都确保以上所言,是件困难的事,但当你考虑到异常、函数内多重回传路径、程序维护员改动软件却没能充分理解随之而来的冲击,态势就很明显了:资源管理的特殊手段还不很充分够用。

【条款13】以对象管理资源

为防止资源泄露,请使用RAII对象,它们在构造函数中获得资源并在析构函数中释放资源。
两个常被使用的RAII classes分别是tr1::shared_ptr和auto_ptr。前者通常是较佳选择,因为其copy行为比较直观。若选择auto_ptr,复制动作会使它(被复制物)指向null。

【条款14】在资源管理类中小心copying行为

条款13适合管理heap-based资源,而并非所有资源都是heap-based,比如互斥器,此时像auto_ptr和tr1::shared_ptr这样的智能指针往往不适合作为资源掌管者。因此,你需要建立自己的资源管理类。
void lock(Mutex* pm);
void unlock(Mutex* pm);

class Lock {
public:
     explicit Lock(Mutex* pm)
     : mutexPtr(pm)
     { lock(mutexPtr); }
     ~Lock() { unlock(mutexPtr); }
private:
     Mutex *mutexPtr;
};
复制RAII对象必须一并赋值它所管理的资源,所以资源的copying行为决定RAII对象的copying行为。
普遍而常见的RAII class copying行为是:抑制copying、施行引用计数法。不过其他行为也都可能被实现。
class Lock: private Uncopyable {
public:
     ...
};

class Lock {
public:
     explicit Lock(Mutex* pm)
     : mutexPtr(pm, unlock)
     {
          lock(mutexPtr.get());
     }
private:
     std::tr1::shared_ptr<Mutex> mutexPtr;
};

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

APIs往往要求访问原始资源,所以每一个RAII class应该提供一个“取得其所管理之资源”的办法。
class Investment {
public:
     bool isTaxFree() const;
     ...
};

std::tr1::shared_ptr<Investment> pInv(createInvestment());
int daysHeld(const Investment* pi);

int days = daysHeld(pInv.get());

bool taxable = !(pInv->isTaxFree());
对原始资源的访问可能经由显式转换或隐式转换。一般而言显式转换比较安全,但隐式转换对客户比较方便。
FontHandle getFont();
void releaseFont(FontHandle fh);

// RAII class
class Font {
public:
     explicit Font(FontHandle fh)
     : f(fh)
     { }
     ~Font() { releaseFont(f); }

     FontHandle get() const { return f; } // implicitly

     operator FontHandle() const // explicitly
     { return f; }
private:
     FontHandle f;
};

void changeFontSize(FontHandle f, int newSize);
Font f(getFont());
int newFontSize;
...
changeFontSize(f.get(), newFontSize); // implicitly change Font to FontHandle

changeFontSize(f, newFontSize); // explicitly change Font to FontHandle
// may be dangerous
Font f1(getFont());
...
FontHandle f2 = f1; // oops, after f1 is destroyed, f2 becomes dangle.

【条款16】成对使用new和delete时要采用相同形式

如果你在new表达式中使用[],必须在相应的delete表达式中也是用[]。如果你在new表达式中不使用[],一定不要在相应的delete表达式中使用[]。

【条款17】以独立语句将newed对象置入智能指针

以独立语句将newed语句存储入(置入)智能指针内。如果不这样做,一旦异常被抛出,有可能导致难以察觉的资源泄露。
int priority();
void processWidget(std::tr1::shared_ptr<Widget> pw, int priority);

// not suggested, case
// first step: new Widget,
// second step: invoke priority(), if something goes wrong, resulting resource leak.
processWidget(std::str1::shared_ptr<Widget>(new Widget), priority());

// suggested
std::str1::shared_ptr<Widget> pw(new Widget);
processWidget(pw, priority());


参考:Effective C++:改善程序与设计的55个具体做法(第三版) Scott Meyers著 侯捷译

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值