C 箴言将new出来的对象存入智能指针解析

  1. 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
  2. 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
  3. 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。

C++箴言:将new出来的对象存入智能指针

不要忘记使用对象管理资源的至理名言,processWidget 为处理动态分配的Widget 使用了一个智能指针。

假设我们有一个函数取得我们的处理优先级,而第二个函数根据优先级针

Widget 使用了一个智能指针(在此,是一个 tr1::shared_ptr)。

个裸指针(raw pointer)应该是显式的,所以不能从一个由 "new Widget" 返回的裸指针隐式转型到 processWidget 所需要的 tr1::shared_ptr.下面的代

还是可能泄漏资源。下面就来说明这是如何发生的。

在编译器能生成一个对 processWidget 的调用之前,它们必须传递实际参数来计算形式参数的值。第二个实际参数不过是对函数 priority 的调用,但是第一个实际参数("std::tr1::shared_ptr(new Widget)"),由两部分组成:

·表达式 "new Widget" 的执行。

·一个对 tr1::shared_ptr 的构造函数的调用。

在 processWidget 能被调用之前,编译器必须为这三件事情生成代码:·调用 priority。

·执行 "new Widget"。

·调用 tr1::shared_ptr 的构造函数。

C++编译器允许在一个相当大的范围内决定这三件事被完成的顺序。(这里与Java和C#等语言的处理方式不同,那些语言里函数参数总是按照一个精确的顺序被计算。)"new Widget" 表达式一定在 tr1::shared_ptr 的构造函数能被调用之前执行,因为这个表达式的结果要作为一个参数传递给

tr1::shared_ptr 的构造函数,但是 priority 的调用可以被第一个,第二个或第三个执行。如果编译器选择第二个执行它(大概这样能使它们生成更有效率的代码),我们最终得到这样一个操作顺序:

·执行 "new Widget"。

·调用 priority。

·调用 tr1::shared_ptr 的构造函数。

但是请考虑,如果对 priority 的调用引发一个异常将发生什么。在这种情况下,从 "new Widget" 返回的指针被丢失,因为它没有被存入我们期望能阻止

资源泄漏的 tr1::shared_ptr。由于一个异常可能插入资源创建的时间和将资源交给一个资源管理对象的时间之间,所以调用 processWidget 可能会发生一次泄漏。避免类似问题的方法很简单:用一个单独的语句创建 Widget 并将它存

在一个语句之内要小得多。"new Widget" 表达式和 tr1::shared_ptr 的构造函数的调用与 priority 的调用在不同的语句中,所以编译器不会允许 priority 的调用插入它们中间。

Things to Remember

·在一个独立的语句中将 new 出来的对象存入智能指针。如果疏忽了这一点,当异常发生时,可能引起微妙的资源泄漏。

相关文档
最新文档