Effective C++: 03资源管理
所谓资源,就是一旦用了它,将来必须还给系统。C++中的资源有:内存、文件描述符、互斥锁、数据库连接、网络socket等。
13:以对象管理资源
1:像下面这个函数:
void f()
{
Investment *pInv = createInvestment();
...
delete pInv;
}
这个函数中,会有若干情况的发生导致无法执行delete语句,比如”...”内可能有一个过早的return语句;或者createInvestment和delete位于某个循环内,而该循环由于某个continue或goto语句过早退出;或者”...”内的语句抛出了异常。
2:为确保createInvestment返回的资源总是被释放,我们需要将资源放进对象内,当控制流离开f,该对象的析构函数会自动释放那些资源。
把资源放进对象内,我们便可倚赖C++的“析构函数自动调用机制”确保资源被释放。比如使用auto_ptr智能指针(auto_ptr已过时,这里仅拿它举例而已):
void f()
{
std::auto_ptr<Investment> pInv(createInvestment());
...
}
3:new以对象管理资源,有两个关键点:
A:获得资源后立即放进管理对象:“以对象管理资源”的观念常被称为“资源取得时机便是初始化时机”(Resource Acquisition Is Initialization, RAII),因为我们几乎总是在获得一笔资源后于同一语句内以它初始化某个管理对象。有时候获得的资源被拿来赋值某个管理对象,但不论哪一种做法,每一笔资源都在获得的同时立刻被放进管理对象中。
B:管理对象运用析构函数确保资源被释放:不论控制流如何离开区块,一旦对象被销毁,其析构函数自然会被自动调
4:如果你打算手工释放资源,容易发生某些错误。罐装式的资源管理类如std::shared_ptr往往比较能够轻松遵循本条款忠告,但有时候你所使用的资源是目前这些预制式classes无法妥善管理的。既然如此就需要精巧制作你自己的资源管理类。
14:在资源管理类中小心copying行为
每一位RAII class作者一定会面对这样一个问题:“当一个RAII对象被复制,会发生什么事?”大多数时候你会选择以下几种可能:
1:禁止复制;
2:对底层资源使用引用计数法,就像std::shared_ptr那样。通常只要内涵一个std::shared_ptr成员变量,RAII classes便可实现出reference-counting copying行为;
3:复制底层资源;
4:转移底层资源的拥有权,就像std::auto_ptr那样;
15:在资源管理类中提供对原始资源的访问
有时候需要一个函数可将RAII class对象转换为其所含的原始数据,比如std::shared_ptr中的get成员函数,用来执行显示转换,也就是它会返回智能指针内部的原始指针;而且std::shared_ptr也重载了operaotr*和operator->函数,允许隐式转换至底部原始指针。
有些RAII class的作者会提供一个隐式转换函数,用于将RAII对象转换为原始资源,这通常不是一个好的做法。
16:成对使用new和delete时要采取相同形式
1:数组所用的内存通常还包括“数组大小”的记录,以便delete知道需要调用多少次析构函数。单一对象的内存则没有这笔记录。可以把两种不同的内存布局想象如下,其中n是数组大小:

因此:如果你调用new时使用[],你必须在对应调用delete时也使用[]。如果你调用new时没有使用[],那么也不该在对应调用delete时使用[]。违反上面的规则化,将会引起未定义行为。
要注意typedef数组的情况:
typedef std::string AddressLines[];
std::string *pal = new AddressLines;
上面的AddressLines是个数组,尽管这里表面上使用new,实际上是用了new[],所以,delete时需要delete[]:
delete pal; // 未定义
delete [] pal; // fine
为了避免上面的错误,最好尽量不要对数组形式做typedef。
17:以独立语句将new对象置入智能指针
像下面这样的函数调用,也有可能泄漏资源:
processWidget(std::shared_ptr<Widget>(new Widget), priority());
原因在于:在调用processWidget之前,编译器必须创建代码,做以下三件事:调用priority;执行”new Widget”;调用std::shared_pt构造函数。
C++没有规定编译器必须以怎样的次序完成这三件事,有可能是以下面的次序:执行”new Widget”;调用priority;调用std::shared_pt构造函数。
这种情况下,万一对priority的调用导致异常,此时”new Widget”返回的指针将会丢失,从而造成资源泄漏。这种情况的发生是因为:在“资源被创建”和“资源被转换为资源管理对象”两个时间点之间有可能发生异常。
为了避免这种情况的发生,使用分离语句:
std::shared_ptr<Widget> pw(new Widget); processWidget(pw, priority());
Effective C++: 03资源管理的更多相关文章
- 《Effective C++》资源管理:条款13-条款17
		
条款13:以对象管理资源 为了防止资源泄漏,请使用RAII(Resource Acquisition Is Initialization)对象,在构造函数里面获得资源,在析构函数里面释放资源 auto ...
 - 【Effective C++】资源管理
		
资源:动态分配的内存.文件描述器.互斥锁.图形界面中的字型与笔刷.数据库连接以及网络sockets等,无论哪一种资源,重要的是,当你不再使用它时,必须将它还给系统. 条款13:以对象管理资源 当我们向 ...
 - Effective Java 03 Enforce the singleton property with a private constructor or an enum type
		
Principle When implement the singleton pattern please decorate the INSTANCE field with "static ...
 - Effective C++ 3.资源管理
		
//条款13:以对象管理资源 // 1.C++程序中最常使用的资源就是动态分配内存,并且还包括文件描述器,互斥锁,GDI对象.数据库连接.网络socket等.不管哪一种资源,当不再使用的时候必须将其归 ...
 - effective c++:资源管理
		
对象管理资源 createInvestment 函数作用时创建一个invest对象: void f() { Investment *pInv = createInvestment(); // call ...
 - const 修饰成员函数 前后用法(effective c++ 03)
		
目录 const在函数后面 const修饰成员函数的两个作用 const在函数前面 总结 const在函数后面 类的成员函数后面加 const,表明这个函数不会对这个类对象的数据成员(准确地说是非静态 ...
 - More Effective C++: 03异常
		
C++的异常机制使得程序付出某些代价:资源泄漏的可能性增加了:写出具有你希望的行为的构造函数与析构函数变得更加困难:执行程序和库程序尺寸增加了,同时运行速度降低了等等. 但是为什么使用异常呢?C程序使 ...
 - Effective Java Index
		
Hi guys, I am happy to tell you that I am moving to the open source world. And Java is the 1st langu ...
 - Effective C++笔记03:资源管理
		
资源:动态分配的内存.文件描写叙述器.相互排斥锁.图形界面中的字型与笔刷.数据库连接以及网络sockets等,不管哪一种资源,重要的是,当你不再使用它时,必须将它还给系统. 条款13:以对象管理资源 ...
 
随机推荐
- UVA11107 Life Forms SA模板
			
Life Forms Time Limit: 5000MS Memory Limit: 65536K Total Submissions: 16827 Accepted: 4943 Descr ...
 - Ubuntu下安装Libpcap
			
Libpcap是 Unix/Linux 平台下的网络数据捕获函数包,百度百科是这么说的,唉,不管什么来头,只要帮我完成作业就行,安装过程记录如下: 还是那个套路,先在网上搜了一把,大概也就那样,被疯狂 ...
 - day38 01-Spring框架的概
			
Action里面调Service,Service里面调DAO,在Action里面new一个Service,在Service里面new一个DAO.有了Spring之后可以不用new对象了.AOP里面有很 ...
 - day36 05-Hibernate检索方式:离线条件查询
			
图二 离线条件查询 Struts 2是web层的框架.session得在dao层才有.有的时候这些数据是没在你的对象里面的.像是否上传简历.这个字段不在我的数据库里面.例如是否上传简历这个条件不在我们 ...
 - Leetcode686.Repeated String Match重复叠加字符串匹配
			
给定两个字符串 A 和 B, 寻找重复叠加字符串A的最小次数,使得字符串B成为叠加后的字符串A的子串,如果不存在则返回 -1. 举个例子,A = "abcd",B = " ...
 - ue4 fstring 和std::string互转
			
https://forums.unrealengine.com/development-discussion/c-gameplay-programming/6517-convert-std-strin ...
 - PHP--y2k38的解决方法已经时间格式的常用转换
			
y2k38又名千年虫问题,又称Uinx Millennium Bug,此漏洞将会影响到所有32位系统下用Unix时间戳整数来记录时间的PHP,及其它编程语言. 一个整型的变量所能保存的最大时间为203 ...
 - 高效整洁CSS代码原则(上)
			
CSS学起来并不难,但在大型项目中,就变得难以管理,特别是不同的人在CSS书写风格上稍有不同,团队上就更加难以沟通,为此总结了一些如何实现高效整洁的CSS代码原则: 1. 使用Reset但并非全局Re ...
 - spring boot 异常处理(转)
			
spring boot在异常的处理中,默认实现了一个EmbeddedServletContainerCustomizer并定义了一个错误页面到"/error"中,在ErrorMvc ...
 - DirectX11笔记(三)--Direct3D初始化代码
			
原文:DirectX11笔记(三)--Direct3D初始化代码 版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/u010333737/article ...