【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 过滤中使用位操作 问题 你想在查询的过滤条件中使用位操作. 解决方案 假 ...
随机推荐
- ASP数据库连接方法语法总结
经常使用到有关数据库的操作.包括连接代码,SQL命令等等,又不曾刻意去记忆它们(我本人是不愿意去记这东东),所以常常在用到的时候又去查书本,翻来翻去.一些比较少用的数据库还不一定能顺利找到,所以现在把 ...
- Python编码相关
1.#coding=utf-8的作用 作用是这个文件代码的编码格式,如果没有声明代码中不能出现中文字符,包括注释中也不能出现.否则会报错SyntaxError: Non-ASCII character ...
- 【Jmeter Linux环境下运行方法】
Jmeter 运行 1.cd jmeter/apache-jmeter-4.0/bin 2.执行 ./jmeter -n -t jmx脚本文件 -l 测结果.jtl文件 -e -o html文件路径 ...
- Service Cloud 零基础(二)Knowledge浅谈
本篇参考:https://trailhead.salesforce.com/content/learn/projects/set-up-salesforce-knowledge https://tra ...
- Security篇:RememberMe
RememberMe功能 rememberMeServices接口 有AbstractRememberMeServices抽象类 PersistentTokenBasedRememberMeServi ...
- Unity状态机(Animator)
状态机的状态(State) 每个Animator Controller都会自带三个状态:Any State, Entry和 Exit.
- IO(02)--属性集、缓冲流、转换流
属性集[Properties] java.util.Properties类继承于Hashtable,用来表示一个持久的属性集.它使用键值结构存储数据,每个键及其对应的值都是一个字符串. 构造方法 ...
- js 获取某月第一天和最后一天
1.获取某月第一天和最后一天日期 function getDateByMonth (timeStamp) { let inDate = new Date(timeStamp) let year = i ...
- k8s第二回之k8s集群的安装
1. k8s集群的安装 目录 1. k8s集群的安装 1.架构: 2.环境准备 3.master节点安装etcd 4. master节点安装kubernetes 5.node节点安装kubernete ...
- aaencode:用颜文字来加密吧
今天逛大佬博客发现了一个有意思的东西 ゚ω゚ノ= /`m´)ノ ~┻━┻ //*´∇`*/ ['_']; o=(゚ー゚) =_=3; c=(゚Θ゚) =(゚ー゚)-(゚ー゚); (゚Д゚) =(゚Θ゚ ...