1. 可能会出现资源泄漏的一种用法

假设我们有一个获取进程优先权的函数,还有一个在动态分配的Widget对象上根据进程优先权进行一些操作的函数:

 int priority();

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

注意这里使用了对象管理资源的用法(Item 13),processWidget为它需要处理的动态分配对象Widget使用了智能指针(tr1::shared_ptr)。

现在考虑对processWidget函数的调用:

 processWidget(new Widget, priority());

这个函数调用不能通过编译,因为在tr1::shared_ptr构造函数中显示的使用了一个原生指针,而不能将“new Widget”返回的原生指针直接隐式转换为tr1::shared_ptr。下面的代码将会通过编译:

 processWidget(std::tr1::shared_ptr<Widget>(new Widget), priority());

虽然这里我们使用了对象类管理资源,但是这个调用仍然可能出现内存泄漏。如何才能出现资源泄漏呢?

2. 在什么情况下会出现资源泄漏?

在编译器生成一个对processWidget的调用之前,它们必须对函数参数做一些检查。第二个参数只是调用了函数priority,但是第一个参数包含两部分:

  • 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中,但我们的原意是使用tr1::shared_ptr来防止资源泄漏。对processWidget的调用会使资源泄漏发生,因为在资源被创建和将资源转交给资源管理对象的时间间隔内插入了异常。

3. 如何避免资源泄漏

防止这个问题的方法比较简单:使用一个单独的句子创建Widget并将其存入智能指针,然后将智能指针传入processWidget:

 std::tr1::shared_ptr<Widget> pw(new Widget); // store newed object

 // in a smart pointer in a

 // standalone statement

 processWidget(pw, priority()); // this call won’t leak

这种方法是行得通的,编译器被给予更少的余地来对语句进行重新排序。在上面的代码中,我们将“new Widget”以及对tr1::shared_ptr构造函数的调用放在一个语句中,把对priority的调用放在另一个语句中,这样就不允许编译器在”new Priority”和tr1::shared_ptr构造函数之间执行priority。

4. 总结

在智能指针中存储new出来的对象时要用单独的语句,不然抛出异常的时候会发生微妙的资源泄漏。

读书笔记 effective c++ Item 17 使用单独语句将new出来的对象放入智能指针的更多相关文章

  1. EC笔记:第三部分:17、使用独立的语句将newed对象放入智能指针

    一般的智能指针都是通过一个普通指针来初始化,所以很容易写出以下的代码: #include <iostream> using namespace std; int func1(){ //返回 ...

  2. 读书笔记_Effective_C++_条款十七:以独立语句将new产生的对象置入智能指针

    int get_int(); void f(shared_ptr<int> a, int); //下面调用 f(new int(3), get_int());//如果是类而不是int就可以 ...

  3. 读书笔记 effective c++ Item 13 用对象来管理资源

    1.不要手动释放从函数返回的堆资源 假设你正在处理一个模拟Investment的程序库,不同的Investmetn类型从Investment基类继承而来, class Investment { ... ...

  4. Effective C++(17) 以独立语句将newed对象置入智能指针

    问题聚焦:     使用了资源管理对象(如智能指针),就一定是安全的吗?显然不是.     资源泄露发生可能在于,在“资源被创建”和“资源被转换为资源管理对象”两个时间点之间有可能发生异常干扰. 看下 ...

  5. 读书笔记 effective c++ Item 29 为异常安全的代码而努力

    异常安全在某种意义上来说就像怀孕...但是稍微想一想.在没有求婚之前我们不能真正的讨论生殖问题. 假设我们有一个表示GUI菜单的类,这个GUI菜单有背景图片.这个类将被使用在多线程环境中,所以需要mu ...

  6. 读书笔记 effective c++ Item 31 把文件之间的编译依赖降到最低

    1. 牵一发而动全身 现在开始进入你的C++程序,你对你的类实现做了一个很小的改动.注意,不是接口,只是实现:一个私有的stuff.然后你需要rebuild你的程序,计算着这个build应该几秒钟就足 ...

  7. 读书笔记 effective c++ Item 36 永远不要重新定义继承而来的非虚函数

    1. 为什么不要重新定义继承而来的非虚函数——实际论证 假设我告诉你一个类D public继承类B,在类B中定义了一个public成员函数mf.Mf的参数和返回类型并不重要,所以假设它们都是void. ...

  8. 读书笔记 effective c++ Item 54 让你自己熟悉包括TR1在内的标准库

    1. C++0x的历史渊源 C++标准——也就是定义语言的文档和程序库——在1998被批准.在2003年,一个小的“修复bug”版本被发布.然而标准委员会仍然在继续他们的工作,一个“2.0版本”的C+ ...

  9. 读书笔记 effective c++ Item 52 如果你实现了placement new,你也要实现placement delete

    1. 调用普通版本的operator new抛出异常会发生什么? Placement new和placement delete不是C++动物园中最常遇到的猛兽,所以你不用担心你对它们不熟悉.当你像下面 ...

随机推荐

  1. 一个基于ATMEGA128的直流电机抱死程序(转)

    源:一个基于ATMEGA128的直流电机抱死程序 先说一下我的硬件情况:一块ATMEGA128实验板:一个带编码器的80:1的变速电机,编码器的输出端连接到单片机的PD4和PD5引脚:一块电机驱动电路 ...

  2. Linux之文件过滤分割与合并

    文件过滤分割与合并 1.grep命令 grep(global regular expression print)全面搜索正则表达式并把行打印出来,是一种强大的文本搜索工具,它能使用正则表达式搜索文本, ...

  3. Beautiful Soup 定位指南

    Reference: http://blog.csdn.net/abclixu123/article/details/38502993 网页中有用的信息通常存在于网页中的文本或各种不同标签的属性值,为 ...

  4. iOS UITabBar

    参考文章:http://www.cnblogs.com/wendingding/p/3775488.html 简单明了,不用再总结了.

  5. java解析XML,并生成文档

    一.Java中XML的四种方法 Java学习者须知:Java中XML的四种方法 java xml学习总结(4中方法的例子介绍) JDOM解析XML Dom4j解析XML

  6. Javascript中alert</script>的方法

    Javascript中alert</script>的方法: <%@ page language="java" import="java.util.*&q ...

  7. Chrome控制台详解

    Chrome控制台详解 http://www.codeceo.com/article/chrome-console.html console.log('%casdf','font-size:16px; ...

  8. 用 NodeJS 实现 BigPipe

    BigPipe 是 Facebook 开发的优化网页加载速度的技术.网上几乎没有用 node.js 实现的文章,实际上,不止于 node.js,BigPipe 用其他语言的实现在网上都很少见.以至于这 ...

  9. Netty学习笔记

    一些类与方法说明 1)ByteBuf ByteBuf的API说明: Creation of a buffer It is recommended to create a new buffer usin ...

  10. doubango(2)--底层协议栈结构分析

    tsip_stack_handle_t 实例 1.        tsip_stack_handle_t的创建 在底层,真正运转的协议栈结构式tsip_stack_handle_t的一个实例,它的创建 ...