Effective C++ 笔记(三)

三:资源管理
条款13:以对象管理资源
以对象管理资源的两个关键想法:
①获得资源后立刻放进管理对象内。
②管理对象运用析构函数确保资源被释放
把资源放进对象内,我们便可以依赖C++的“析构函数自动调用机制”确保资源被释放。如auto_ptr和shared_ptr智能指针。
auto_ptr,复制动作会使它(被复制物)指向null。
为防止资源泄露,请使用 RAII(Resource Acquisition Is Initialization)“资源取得时机便是初始化时机”。

//错误版本!!!
void f(){
	Investment *pInv = createInvestment();//调用factory函数
	...                                    //如果这中间有个return被执行,就不会执行delete而造成资源泄露。
	delete pInv;                          //释放pInv所指对象
}

//正确版本
void f(){
	...
	std::tr1::shared_ptr<Investment>
	pInv(createInvestment());  //调用factory函数
	...						  //使用pInv
								//经由shared_ptr析构函数自动删除pInv
}

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

条款14:在资源管理类中小心copying行为
复制底部资源:可以针对一份资源拥有其任意数量的复件(副本)。当不在需要某个复件时确保他被释放。在此情况下复制资源管理对象,应该同时复制其所包覆的资源,也就是说,复制资源管理对象时,进行的是“深度拷贝”。(p68)

请记住:1.复制RAII对象必须一并复制他所管理的资源,所以资源的copying行为决定RAII对象的copying行为。
2.普遍而常见的RAII class copying 行为是:抑制copying、施行引用计数法。

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

std::tr1::shared_ptr<Investment> pInv(createInvestment());
//如果想用某个函数处理Investment对象:
int daysHeld(const Investment*pi);  //返回原始资源
//如果这样调用,错误
int days = daysHeld(pInv);//错误,因为pInv类型是shared_ptr<Investment>而不是Investment
//可以这样调用
int days = daysHeld(pInv.get());

shared_ptr和auto_ptr都提供了一个get成员函数,用来执行显示转换,也就是他会返回智能指针内部的原始指针(的复件)。
就像几乎所有智能指针一样,shared_ptr和auto_ptr也重载了operator ->和operator *,允许隐式转换至底部原始指针。p70

请记住:1.API往往要求访问原始资源,所以每一个RAII class应该提供一个“取得其所管理之资源”的办法。
2.对原始资源的访问可能经由显示转换或隐式转换(p71)。一般而言显示转换比较安全,但隐式转换对客户比较方便。

条款16:成对使用new和delete时要采取相同形式
请记住:如果你在new表达式中使用[],必须在相应的delete表达式中也使用[]。如果你在new表达式中不使用[],一定也不要在delete表达式中使用[]。

条款17:以独立语句将newed对象置于智能指针

int priority();
void processWidget(std::tr1::shared_ptr<Widget> pw,int priority);

//调用processWidget
processWidget(new Widget,priority()); //错误,不通过编译,第一个参数类型不匹配,并且该构造函数十个explicit,无法进行隐式转换。

/*
下面这个通过编译
	但可能导致资源泄露
	C++编译器 以什么次序完成 调用priority,执行new Widget,调用shared_ptr构造函数, 弹性很大,priority的调用排在第几不明确,
	如果priority排在第二,执行new Widget排在第一,调用shared_ptr构造函数最后,并且priority的调用导致异常,那么new Widget尚未被置								入shared_ptr,就会引发资源泄露		
*/
processWidget(std::tr1::shared_ptr<Widget> (new Widget),priority());

//正确版本
std::tr1::shared_ptr<Widget> pw (new Widget);
processWidget(pw,priority());

请记住:以独立语句将newed对象存储于(置入)智能指针内。如果不这样做,一旦异常被抛出,有可能导致难以察觉的资源泄露。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

二零二三.

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值