用RAII技术管理资源及其泛型实现

2024-05-09 14:08

本文主要是介绍用RAII技术管理资源及其泛型实现,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!


前言

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技术管理资源及其泛型实现的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/973607

相关文章

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

综合安防管理平台LntonAIServer视频监控汇聚抖动检测算法优势

LntonAIServer视频质量诊断功能中的抖动检测是一个专门针对视频稳定性进行分析的功能。抖动通常是指视频帧之间的不必要运动,这种运动可能是由于摄像机的移动、传输中的错误或编解码问题导致的。抖动检测对于确保视频内容的平滑性和观看体验至关重要。 优势 1. 提高图像质量 - 清晰度提升:减少抖动,提高图像的清晰度和细节表现力,使得监控画面更加真实可信。 - 细节增强:在低光条件下,抖

【专题】2024飞行汽车技术全景报告合集PDF分享(附原数据表)

原文链接: https://tecdat.cn/?p=37628 6月16日,小鹏汇天旅航者X2在北京大兴国际机场临空经济区完成首飞,这也是小鹏汇天的产品在京津冀地区进行的首次飞行。小鹏汇天方面还表示,公司准备量产,并计划今年四季度开启预售小鹏汇天分体式飞行汽车,探索分体式飞行汽车城际通勤。阅读原文,获取专题报告合集全文,解锁文末271份飞行汽车相关行业研究报告。 据悉,业内人士对飞行汽车行业

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

让树莓派智能语音助手实现定时提醒功能

最初的时候是想直接在rasa 的chatbot上实现,因为rasa本身是带有remindschedule模块的。不过经过一番折腾后,忽然发现,chatbot上实现的定时,语音助手不一定会有响应。因为,我目前语音助手的代码设置了长时间无应答会结束对话,这样一来,chatbot定时提醒的触发就不会被语音助手获悉。那怎么让语音助手也具有定时提醒功能呢? 我最后选择的方法是用threading.Time

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

C#实战|大乐透选号器[6]:实现实时显示已选择的红蓝球数量

哈喽,你好啊,我是雷工。 关于大乐透选号器在前面已经记录了5篇笔记,这是第6篇; 接下来实现实时显示当前选中红球数量,蓝球数量; 以下为练习笔记。 01 效果演示 当选择和取消选择红球或蓝球时,在对应的位置显示实时已选择的红球、蓝球的数量; 02 标签名称 分别设置Label标签名称为:lblRedCount、lblBlueCount

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略 1. 特权模式限制2. 宿主机资源隔离3. 用户和组管理4. 权限提升控制5. SELinux配置 💖The Begin💖点点关注,收藏不迷路💖 Kubernetes的PodSecurityPolicy(PSP)是一个关键的安全特性,它在Pod创建之前实施安全策略,确保P

金融业开源技术 术语

金融业开源技术  术语 1  范围 本文件界定了金融业开源技术的常用术语。 本文件适用于金融业中涉及开源技术的相关标准及规范性文件制定和信息沟通等活动。