【C++】《Effective C++》第三章
第三章 资源管理
条款13:以对象管理资源
当申请一块动态内存时,可能会发生内存泄漏。
class Investment {};
void f() {
Investment* pInv = createInvestment();
// ...
delete pInv; // 释放pInv所指对象
}
对于以上程序,发生内存泄漏的情况:
- 忘记
delete。 - 有
delete,但是在delete之前就跳出了控制流。比如提前return、发生异常等。
- 忘记
解决这种情况比较好的方法是使用对象管理资源,它包括两个关键想法:
- 获得资源后立刻放进管理对象内,即资源取得时机便是初始化时机(
RAII)。 - 管理对象运用析构函数确保资源被释放。
- 获得资源后立刻放进管理对象内,即资源取得时机便是初始化时机(
#include <memory>
// 使用智能指针auto_ptr,C++11中使用unique_ptr,auto_ptr是其老版本。
void f() {
std::auto_ptr<Investment> pInv(createInvestment());
// std::unique_ptr<Investment> pInv(createInvestment());
// ...
}
// 使用智能指针shared_ptr,推荐使用
void f() {
std::shared_ptr<Investment> pInv = std::make_shared<Investment>(createInvestment());
}
请记住
- 为防止资源泄漏,请使用
RAII对象,它们在构造函数中获得资源并在析构函数中释放资源。 - 常用的
RAII classes有unique_ptr,auto_ptr,shared_ptr,推荐使用shared_ptr。
条款14:在资源管理类中小心copying行为
并非所有资源都是动态内存,除此之外还有锁等资源,也应该通过"对象管理资源"来确保获取资源后能够正确的释放,根据资源的类型和不同的需求,可能需要定义不同的copy行为。
- 禁止复制:比如说锁资源,管理锁资源的对象复制通常并不合理。因此应该禁止这类对象的复制。
- 对底层资源使用"引用计数法":如果希望保有资源,直到它的最后一个使用者(某对象)被销毁。这种情况下复制
RAII对象时,应该将资源的"被引用数"递增。 - 复制底部资源:这种情况下,希望在复制
RAII对象时,同时复制其关联的底层资源。展现出一种"深拷贝"行为。 - 转移底部资源的拥有:如果希望任一时刻一个资源只由一个
RAII对象管理,那么在复制RAII对象时,应该实现拥有权的"转移",原RAII对象拥有的资源设为null。
请记住
- 复制
RAII对象必须一并复制它所管理的资源,所以资源的copying行为决定RAII对象的copying行为。 - 普遍而常见的
RAII class copying行为是:抑制copying、施行引用计数法(reference counting)。不过其他行为也都可能被实现。
条款15:在资源管理类中提供对原始资源的访问
取得RAII对象所管理资源的办法可以通过显式转换或隐式转换:
- 显式转换(比较安全,但不易用):如
shared_ptr的get()方法。 - 隐式转换(比较易用,但不安全):如
shared_ptr的operator*和operator->。
请记住
- APIs往往要求访问原始资源。所以每一个RAII class应该提供一个"取得其所管理之资源"的办法。
- 对原始资源的访问可能经由显式转换和隐式转换。一般而言显式转换比较安全,但隐式转换对客户比较方便。
条款16:成对使用new和delete时要采取相同形式
当使用new和delete时,发生两件事:
- new:
- 内存被分配出来(通过
operator new函数)。 - 针对此内存会有一个(或更多)构造函数被调用。
- 内存被分配出来(通过
- delete:
- 针对此内存会有一个(或更多)析构函数被调用。
- 内存被释放(通过
operator delete函数)。
std::string* stringPtr1 = new std::string;
std::string* stringPtr2 = new std::string[100];
// ...
delete stringPtr1;; // 删除一个对象
delete [] stringPtr2; // 删除一个由对象组成的数组
请记住
- 如果你在
new表达式中使用[],必须在相应的delete表达式中也使用[]。如果你在new表达式中不使用[],一定不要在相应的delete表达式中使用[]。
条款17:以独立语句将newed对象置入智能指针
// 揭示处理程序的优先权
int priority();
void processWidget(std::shared_ptr<Widget> pw, int priority);
// 调用
processWidget(new Widget, priority());
// 显然上面的调用不对
processWidget(std::shared_ptr<Widget>(new Widget), priority());
虽然如此,上面的调用还是可能泄露资源。因为在调用processWidget之前,编译器必须创建代码,做以下三件事:调用priority;执行new Widget;调用std::shared_ptr构造函数。但是在C++中编译器对于这三件事的调用顺序不定,所以当priority的调用导致异常,可能执行new Widget返回的指针将会遗失。即在资源被创建和资源被转换为资源管理对象两个时间点之间可能发生异常干扰。
为了解决上面的问题,可以使用分离语句。
// 分成两步调用
std::shared_ptr<Widget> pw(new Widget);
processWidget(pw, priority());
请记住
- 以独立语句将
newed对象存储于(置入)智能指针内。如果不这样做,一旦异常被抛出,有可能导致难以察觉的资源泄露。
【C++】《Effective C++》第三章的更多相关文章
- [Effective Java]第三章 对所有对象都通用的方法
声明:原创作品,转载时请注明文章来自SAP师太技术博客( 博/客/园www.cnblogs.com):www.cnblogs.com/jiangzhengjun,并以超链接形式标明文章原始出处,否则将 ...
- 对于所有对象都通用方法的解读(Effective Java 第三章)
这篇博文主要介绍覆盖Object中的方法要注意的事项以及Comparable.compareTo()方法. 一.谨慎覆盖equals()方法 其实平时很少要用到覆盖equals方法的情况,没有什么特殊 ...
- 《Effective Java 第三版》新条目介绍
版权声明:本文为博主原创文章,可以随意转载,不过请加上原文链接. https://blog.csdn.net/u014717036/article/details/80588806前言 从去年的3月份 ...
- 【软件构造】第三章第三节 抽象数据型(ADT)
第三章第三节 抽象数据型(ADT) 3-1节研究了“数据类型”及其特性 ; 3-2节研究了方法和操作的“规约”及其特性:在本节中,我们将数据和操作复合起来,构成ADT,学习ADT的核心特征,以及如何设 ...
- 《Django By Example》第三章 中文 翻译 (个人学习,渣翻)
书籍出处:https://www.packtpub.com/web-development/django-example 原作者:Antonio Melé (译者注:第三章滚烫出炉,大家请不要吐槽文中 ...
- 《Linux内核设计与实现》读书笔记 第三章 进程管理
第三章进程管理 进程是Unix操作系统抽象概念中最基本的一种.我们拥有操作系统就是为了运行用户程序,因此,进程管理就是所有操作系统的心脏所在. 3.1进程 概念: 进程:处于执行期的程序.但不仅局限于 ...
- Python黑帽编程3.0 第三章 网络接口层攻击基础知识
3.0 第三章 网络接口层攻击基础知识 首先还是要提醒各位同学,在学习本章之前,请认真的学习TCP/IP体系结构的相关知识,本系列教程在这方面只会浅尝辄止. 本节简单概述下OSI七层模型和TCP/IP ...
- 《Entity Framework 6 Recipes》中文翻译系列 (11) -----第三章 查询之异步查询
翻译的初衷以及为什么选择<Entity Framework 6 Recipes>来学习,请看本系列开篇 第三章 查询 前一章,我们展示了常见数据库场景的建模方式,本章将向你展示如何查询实体 ...
- 《Entity Framework 6 Recipes》中文翻译系列 (19) -----第三章 查询之使用位操作和多属性连接(join)
翻译的初衷以及为什么选择<Entity Framework 6 Recipes>来学习,请看本系列开篇 3-16 过滤中使用位操作 问题 你想在查询的过滤条件中使用位操作. 解决方案 假 ...
随机推荐
- 原创:DynamicDataDisplay波形显示自定义格式
原创:DynamicDataDisplay 原版本在日期显示的格式上与我们的习惯不一样,特做如下修改: 自定义日期格式修改: //MainWindow.cs中 var ds = new Enumera ...
- 题解-比赛CF1332
题解-比赛CF1332 比赛CF1332 [A] [B] [C] [D] [E] [F] [G] [A]Exercising Walk Exercising Walk \(T\) 组测试数据,每次给定 ...
- springboot配置ssl证书
springboot默认使用的是tomcat: 1.先到阿里云上注册一个证书,绑定域名:后面可以在管理中下载证书,下载tomcat对应的证书(一个*.pfx文件和*.txt文件) 2.将pfx文件拷贝 ...
- 10分钟快速入门vue.js
Vue.js是一个轻巧.高性能.可组件化的MVVM库,一套用于构建用户界面的渐进式框架,上手简单,兼容强大. 官方文档:https://cn.vuejs.org/v2/guide/ 下面我们就直接来使 ...
- Spring AOP的理解(通俗易懂)。
转载 原文链接:http://www.verydemo.com/demo_c143_i20837.html 这种在运行时,动态地将代码切入到类的指定方法.指定位置上的编程思想就是面向切面的编程. 1. ...
- JavaSE04-Switch&循环语句
1.Switch 格式: 1 switch (表达式) { 2 case 1: 3 语句体1; 4 break; 5 case 2: 6 语句体2; 7 break; 8 ... 9 default: ...
- C# 好代码学习笔记(1):文件操作、读取文件、Debug/Trace 类、Conditional条件编译、CLS
目录 1,文件操作 2,读取文件 3,Debug .Trace类 4,条件编译 5,MethodImpl 特性 5,CLSCompliantAttribute 6,必要时自定义类型别名 目录: 1,文 ...
- Unity 黑暗之光 笔记 第三章
第三章 角色控制 1.创建游戏运行场景并导入素材资源 2.创建和管理标签 1 //const 表明这个是一个共有的不可变的变量 2 public const string ground = &qu ...
- Java中多线程安全问题实例分析
案例 1 package com.duyang.thread.basic.basethread; 2 3 /** 4 * @author :jiaolian 5 * @date :Created in ...
- [GXYCTF2019]禁止套娃(无参RCE)
[GXYCTF2019]禁止套娃 1.扫描目录 扫描之后发现git泄漏 使用githack读取泄漏文件 <?php include "flag.php"; echo &quo ...