前言

RAII的含义是“资源获取即初始化”。

一段看似安全的代码

首先看一段代码:

  1. try{
  2. int *p = new int[100];
  3. // ... do something
  4. delete[] p;
  5. }catch(exception &e){
  6. // .....
  7. }

这段代码中,我们先进行了动态内存分配,使用完释放,看起来很完美,但是这段程序是否真的保证不会发生内存泄漏?

考虑这样一种情形,程序在使用这段内存的过程中throw一个异常,于是程序转向catch块,然后XXX。 这段内存被释放了吗? 显然没有。 那么这段程序应该从哪里改进呢?

对象的生命期

考虑一个问题:C++中对象的声明期是怎样的?在C++中,对象创建的方式有两种,一种是栈上,一种是堆上创建。

  1. {
  2. Animal a; // stack
  3. Animal *pa = new Animal(); // heap
  4. }

上面的代码中,第一个对象创建在栈上,更明确的说法是它是一个局部变量,这意味着它的生命期起源于被创建的这行语句,终结与所在作用域的末尾,也就是这里的右花括号(})。
而第二个对象呢?它是采用所谓的动态内存分配生成的,需要程序员手工去释放,当调用delete的时候才销毁,但调用delete的时机不是固定的。
也就是说,栈对象的生命期是明确的,而堆对象的生命期由于取决于调用delete的时机,因而是不明确的。

看到这里,之前那段有可能内存泄漏的代码如何去改进呢?

答案就是用栈对象明确的生命期去管理资源

用对象的生命期管理资源

试想一下,如果我们把之前程序中,对内存的分配写在构造函数中,把释放资源写在析构函数中,而栈对象的生命期是明确的,当该管理资源的对象过期时,连同它管理的资源一起释放,岂不是非常智能化?

我们尝试着写出下列代码:

  1. class ScopePtr{
  2. public:
  3. ScopePtr(int *p):_p(p){
  4. }
  5. ~ScopePtr(){
  6. delete[] _p;
  7. }
  8. private:
  9. int *_p;
  10. };

我们把之前的代码做如下的改进:

  1. try{
  2. ScopePtr scope(new int[100]);
  3. // ... do something
  4. }catch(exception &e){
  5. // .....
  6. }

再来分析一下这段代码:
如果正常执行,那么当执行完try块时,scope对象过期,执行析构函数,同时释放了那段数组。如果使用的过程中发生了异常,那么当程序进入catch块时,同样会销毁try内的局部变量。
无论是哪种情况,内存总是会被释放。
如果这里不是int,而是其他复杂的类型,使用这个封装的ScopePtr是不是不太方便?显然不会,我们去重载成员操作符就可以了,使它表现的像个指针,这就是一个最简单的智能指针的产生。
问题得到了完美的解决!

资源获取即初始化

我们上面解决问题的办法就是RAII技术,RAII的含义是“资源获取即初始化”,这个概念有两个要点:

  • 获得资源后立即放进管理对象
  • 管理对象运用析构函数确保资源被释放

看另外一个例子:我们在访问一些临界区资源的时候通常需要加锁,所以产生了下面的代码:

  1. {
  2. mutex.lock();
  3. //do sth..
  4. mutex.unlock();
  5. }

这种方式是很容易出现问题的,例如程序中间遇见错误情况需要退出这个函数,此时很容易忘记解锁:

  1. {
  2. mutex.lock();
  3. //do sth..
  4. if(...){
  5. return false // forgot to unlock
  6. }
  7. // ...
  8. mutex.unlock();
  9. }

此时如果再次进行Lock操作,就造成了死锁。
解决这个问题的办法仍然很简单,我们去写一个类:

  1. class MutexLockGuard{
  2. public:
  3. MutexLockGuard(MutexLock mutex):_mutex(mutex){
  4. _mutex.lock();
  5. }
  6. ~MutexLockGuard(){
  7. _mutex.unlock();
  8. }
  9. private:
  10. MutexLock &_mutex;
  11. };

这样刚才那段代码就可以修改成:

  1. {
  2. MutexLockGuard guard(lock);
  3. //do sth..
  4. if(...){
  5. return false
  6. }
  7. // ...
  8. }

这样,一旦离开这段代码,程序立刻自动解锁。
不过为了防止错误使用这个类,例如:

  1. MutexLockGuard(lock);

可以定义一个宏:

  1. #define MutexLockGuard(m) "ERR MutexLockGuard"

这样我们在错误使用的时候,编译期间就能发现错误。

一种泛型解决方案

刘未鹏在他的《C++11(及现代C++风格)和快速迭代式开发》中提出了一种泛型实现,利用了C++11的function和Lambda匿名函数,如下:

  1. class ScopeGuard
  2. {
  3. public:
  4. explicit ScopeGuard(std::function<void()> onExitScope)
  5. : onExitScope_(onExitScope), dismissed_(false)
  6. { }
  7. ~ScopeGuard()
  8. {
  9. if(!dismissed_)
  10. {
  11. onExitScope_();
  12. }
  13. }
  14. void Dismiss()
  15. {
  16. dismissed_ = true;
  17. }
  18. private:
  19. std::function<void()> onExitScope_;
  20. bool dismissed_;
  21. private: // noncopyable
  22. ScopeGuard(ScopeGuard const&);
  23. ScopeGuard& operator=(ScopeGuard const&);
  24. };

使用方式也很简单:

  1. HANDLE h = CreateFile(...);
  2. ScopeGuard onExit([&] { CloseHandle(h); });

其实就是将该资源释放的函数代码段注册到Scope类,其中原理不再赘述。

与其他语言的对比

RAII是C++独有的编程手段。通过RAII技术我们能够做到资源不需要使用时立即释放,这是其他GC语言所不具备的。
以Java为例,Java具有完善的GC(Garbage Collection,垃圾回收)机制,但是存在如下的缺点:

  • GC只能回收内存,而对于打开的文件、数据库连接等仍然需要手工关闭。
  • GC因为进程优先级等原因,回收效率底下,详情可以参考孟岩的《垃圾收集机制(Garbage Collection)批判》

conclusion

RAII技术是现代C++编程技术中及其重要的一部分,甚至有人称其为“C++编程中最重要的编程技法”,可见其重要性。通过RAII,我们完全可以实现资源的自动化管理,写出永不内存泄漏的程序。

参考资料

  • 《C++ Primer》
  • 《Effective C++》
  • 《Linux多线程服务器端编程》

用RAII技术管理资源及其泛型实现相关推荐

  1. C++高阶 RAII机制(以对象管理资源)

    文章目录 1.什么是RAII 2.RAII的原理 3.实例: 1.什么是RAII RAII(Resource Acquisition Is Initialization)是由c++之父Bjarne S ...

  2. 0.C++之RAII技术解析

     1.什么是RAII 技术? 我们在C++中经常使用new申请了内存空间,但是却也经常忘记delete回收申请的空间,容易造成内存泄露,于是RAII技术就诞生了,来解决这样的问题.RAII(Res ...

  3. 【RAII】RAII 技术(内存安全解决技术/自动化解锁技术)

    目录 介绍 RAII技术在锁上的应用 互斥锁 RAII技术在指针上的应用 智能指针 介绍 RAII :Resource Acquisition Is Initialisation (RAII) (任何 ...

  4. C++之RAII技术解析

    RAII 是 Resource Acquisition Is Initialization(资源获取即初始化)的缩写.RAII 是C++ 中常用的一种编程技术,具体是指必须在使用前获取的资源(如栈内存 ...

  5. RAII技术与智能指针(超详细)

    RAII技术与智能指针 RAII技术与智能指针 RAII技术 什么是RAII技术 实现原理 特征 作用 值语义与对象语义的区分 RAII代码格式 测试代码(Computer类的实现) 结论 智能指针 ...

  6. 做好技术管理,你必须要跨越的4道槛

    作者:findyi,腾讯.360码农,前哒哒少儿英语技术VP,现任土豆教育CTO. 读者群里有不少刚开始做管理的技术人,很多都和我谈过他们的困惑. 总结下来主要是不知道继续晋升需要培养哪方面的能力. ...

  7. 并不是所有的程序员都适合做技术管理

    摘要:这是曾经的技术高手,如今却是名职业经理人分享的内容.他称,相信很多程序员都想做管理,一方面是高薪,另一方面则是做管理很风光.工作也轻松,但实际上并不是所有人都适合做管理,技术和管理是两完全不同的 ...

  8. 技术管理:带人和团队管理

    声明:所谓的技术管理笔记,是一位原大公司的码农不甘寂寞,出来加入创业公司后的管理心得记录.大公司到创业公司的落差是全方位的,制度,氛围,资源,人才皆有.从最初的不适应到一路磕磕碰碰活到现在.心中充满感 ...

  9. 技术管理规划-从哪入手?

    前言 最近在学习技术管理课程,对学习的内容做一些笔记,结合当前的工作环境在对应的知识点上做一些思考,慢慢建立自己的团队管理方法论. 学习方法 早上骑单车的时候或者跑步的时候听音频,熟悉课程内容: 仔细 ...

最新文章

  1. 机器学习特征表达——日期与时间特征做离散处理(数字到分类的映射),稀疏类分组(相似特征归档),创建虚拟变量(提取新特征) 本质就是要么多变少,或少变多...
  2. 一起谈.NET技术,页面片段缓存(二)
  3. utf-8 字符串转为Unicode编码格式
  4. python生成随机数方法_Python生成随机数的方法
  5. DC概论四之setup time 与 hold time 之三
  6. 【Redis】redis开机自启动、设置守护进程、密码设置、访问权限控制等安全设置(redis默认端口6379)...
  7. @Override must override a superclass method
  8. CSS Sprite精灵图如何缩放大小
  9. python判断字母大写_python判断字符串是字母 数字 大小写(转载)
  10. 几种常见的数据分析模型
  11. 硬件工程师应该知道的音频功放电路
  12. 猿猿趴架构实战--03 什么是架构
  13. VPS云主机怎么用?
  14. 安卓手机续航测试软件,10款手机电池续航测试对决:都是骁龙865手机,续航差距有多大?...
  15. 异常、信息国际化样例
  16. 现代韦编三绝 新东方强浩老师自学新概念的故事
  17. roll() java_java.util.Calendar.roll(int field,int amount)方法实例
  18. 全志D1开发板 XR829蓝牙 Can‘t get device info: No such device 自我分析及解决方案
  19. 香橙派全志H3烧入U-boot和Linux内核以及配置
  20. cad坐标归零lisp_Z轴归零源码说明 - AutoLISP/Visual LISP 编程技术 - CAD论坛 - 明经CAD社区 - Powered by Discuz!...

热门文章

  1. sql between包括两端吗_SQL简单查询
  2. ar面部识别_国内手机厂商AR布局报告
  3. python文件运行哪一个_如何使一个python文件运行另一个?
  4. mysql 支持json_MySQL 5.7 对 JSON 的支持
  5. 计算机系统中存储管理是,《计算机操作系统5、存储管理.doc
  6. 光纤收发器的故障处理
  7. 【渝粤题库】国家开放大学2021春3897商务英语1题目
  8. 别克gl8服务器返回数据为空,老司机教学:别克GL8商务车总线控制系统故障解决...
  9. android封装oauth2,Android AccountAuthenticator和OAuth2
  10. Java集合(1)--集合概述