C++惯用法之RAII思想: 资源管理

C++编程技巧专栏:http://t.csdnimg.cn/eolY7
目录
1.概述
2.RAII的应用
2.1.智能指针
2.2.文件句柄管理
2.3.互斥锁
3.注意事项
3.1.禁止复制
3.2.对底层资源使用引用计数法
3.3.复制底部资源(深拷贝)或者转移资源管理权(移动语义)
4.RAII的优势和挑战
5.总结
1.概述
RAII是Resource Acquisition Is Initialization的缩写,即“资源获取即初始化”。RAII原则的基本思想是将资源的生命周期与对象的生命周期绑定在一起。它是C++语言的一种管理资源、避免资源泄漏的惯用法,利用栈的特点来实现,这一概念最早由Bjarne Stroustrup提出。在函数中由栈管理的临时对象,在函数结束时会自动析构,从而自动释放资源,因此,我们可以通过构造函数获取资源,通过析构函数释放资源。这种自动管理资源的方式可以大大减少资源泄漏、野指针和其他与资源管理相关的问题。常见的写法为:
Object() {// acquire resource in constructor
}
~Object() {// release resource in destructor
}
2.RAII的应用
2.1.智能指针
智能指针是RAII原则在内存管理中的一个典型应用。C++11引入了多种智能指针类型,如std::unique_ptr、std::shared_ptr和std::weak_ptr,它们可以自动管理动态分配的内存。
例如,使用std::unique_ptr可以确保在不需要动态分配的内存时自动释放它:
#include <iostream>
#include <memory>class MyClass {
public:MyClass() { std::cout << "MyClass created\n"; }~MyClass() { std::cout << "MyClass destroyed\n"; }
};int main() {{std::unique_ptr<MyClass> ptr(new MyClass()); // MyClass对象被创建// 当ptr离开这个作用域时,它会自动释放所指向的MyClass对象} // MyClass对象在这里被销毁,输出"MyClass destroyed"return 0;
}
在这个例子中,当ptr离开其作用域时,std::unique_ptr的析构函数会被调用,从而释放它所指向的MyClass对象。这种自动的内存管理方式避免了手动调用delete可能导致的错误。
2.2.文件句柄管理
另一个常见的应用是使用RAII原则管理文件句柄。通过创建一个封装了文件句柄的类,可以确保在不需要文件时自动关闭它。
例如:
#include <fstream>
#include <iostream>class FileWrapper {
public:FileWrapper(const std::string& filename, std::ios_base::openmode mode): file_(filename, mode) {if (!file_.is_open()) {throw std::runtime_error("无法打开文件: " + filename);}}~FileWrapper() {file_.close(); // 在析构函数中关闭文件句柄}// 提供对内部文件的访问(如果需要的话)std::fstream& file() { return file_; }private:std::fstream file_; // 封装文件句柄的成员变量
};
在这个例子中,FileWrapper类的构造函数打开一个文件,并在析构函数中关闭它。这确保了即使在异常情况下,文件句柄也会被正确关闭。
2.3.互斥锁
在多线程编程中,std::lock_guard, std::unique_lock, std::shared_lock等也利用了RAII的原理,用于管理互斥锁。当这些类的等对象创建时,会自动获取互斥锁;当对象销毁时,会自动释放互斥锁。
std::lock_guard的构造函数如下:
template< class Mutex > class lock_guard;
std::lock_guard的析构函数会自动释放互斥锁,因此,我们可以通过std::lock_guard来管理互斥锁,从而避免忘记释放互斥锁。如:
std::mutex mtx;
std::lock_guard<std::mutex> lock(mtx); // unlock when lock is out of scope
不使用RAII的情况下,我们需要手动释放互斥锁,如下所示:
std::mutex mtx;
mtx.lock();
// ...
mtx.unlock();
3.注意事项
在资源管理类中小心copy行为
- 拷贝RAII对象必须考虑其管理的资源,针对其资源做出拷贝行为的实现
- 常见的RAII对象拷贝行为:拒绝拷贝、引用计数法、深拷贝、资源所有权转移
并非所有资源都是基于堆的(heap-based),对于这种对象不能直接使用智能指针,需要自定义其资源管理类。例如:为了说明锁的资源管理行为,我们这里给定义一个锁,来替代C++里的锁
struct MyMutex {MyMutex() {printf("Construct MyMutex\n");}~MyMutex() {printf("Deconstruct MyMutex\n");}
};
其上锁解锁行为:
void lock(MyMutex *) {printf("lock\n");
}void unlock(MyMutex *) {printf("unlock\n");
}
锁的资源管理类,在构造函数获取资源(加锁),在析构函数释放资源(解锁):
struct Lock {
private:MyMutex *myMutex;
public:explicit Lock(MyMutex *mutex) : myMutex(mutex) {lock(myMutex);}~Lock() {unlock(myMutex);}
};
使用:
int main() {MyMutex myMutex;{printf("---------\n");Lock lk(&myMutex);printf("---------\n");// 离开代码块将自动析构局部对象,因此会释放锁}
}
/*
Construct MyMutex
---------
lock
---------
unlock
Deconstruct MyMutex
*/
潜在风险,如果发生了拷贝行为:
Lock l1(&mutex);
Lock l2(l1);
那么将立即死锁(Linux里一般是非递归锁,重复加锁会造成死锁)
3.1.禁止复制
继承nocopyable,或者将拷贝相关函数设置为delete。如:
//[1]
class NonCopyable
{
protected:NonCopyable(const NonCopyable&){}NonCopyable& operator=(NonCopyable&){}
};或//[2]
class NonCopyable
{
public:NonCopyable(const NonCopyable&)=delete;NonCopyable& operator=(const NonCopyable&)=delete;
};
3.2.对底层资源使用引用计数法
思想:维护一个计数器,当最后一个使用者被销毁时,才真正释放资源,如:
struct Lock {
private:shared_ptr<MyMutex> mutexPtr;
public:// 将unlock函数设置为删除器explicit Lock(MyMutex *mutex) : mutexPtr(mutex, unlock) {lock(mutexPtr.get());}// 不必声明析构函数,因为mutexPtr是栈上对象,所以会被默认释放,那么智能指针就会调用其释放器unlock
};
3.3.复制底部资源(深拷贝)或者转移资源管理权(移动语义)
在资源管理类中提供对原始资源的访问
- API常需要要求访问原始资源,所以RAII资源管理类应该提供访问原始资源的接口
- 对原始资源可以由显示转换或者隐式转换获得.其在安全性和方便性上各有取舍
智能指针提供了get接口来访问原始资源
在其中要注意,不可以get一个智能指针去初始化另一个智能指针,否则会发生重复释放
int main() {shared_ptr<MyMutex> p1 = make_shared<MyMutex>();{shared_ptr<MyMutex> p2(p1.get());cout << p1.use_count() << " " << p2.use_count() << endl;
// 1 1
// p2离开代码块,释放其管理的资源,p1指针指向被释放的内存}
}
程序将异常退出
4.RAII的优势和挑战
优势:
-
自动资源管理:通过绑定资源的生命周期与对象的生命周期,RAII自动处理资源的获取和释放,减少了手动管理的错误。
-
代码简洁性:RAII原则鼓励将资源管理逻辑封装在类中,使代码更加清晰和易于维护。
-
异常安全性:当使用RAII时,即使在异常情况下,资源也会被正确释放,这有助于提高程序的健壮性。
挑战:
-
资源所有权的转移:在使用RAII时,需要仔细考虑资源所有权的转移。例如,在使用智能指针时,需要明确何时使用
std::move来转移所有权。 -
与旧代码的兼容性:在将RAII原则应用于现有代码库时,可能需要大量的重构工作来适应新的资源管理方式。
-
学习曲线:对于初学者来说,理解和正确应用RAII原则可能需要一些时间和经验。
5.总结
RAII原则为C++程序员提供了一种强大且优雅的资源管理方法。通过将资源的生命周期与对象的生命周期绑定在一起,RAII不仅简化了资源管理,还提高了代码的健壮性和可维护性。然而,为了充分利用RAII的优势,程序员需要仔细设计类的接口和实现,并考虑到资源所有权和资源转移的问题。
相关文章:
C++惯用法之RAII思想: 资源管理
C编程技巧专栏:http://t.csdnimg.cn/eolY7 目录 1.概述 2.RAII的应用 2.1.智能指针 2.2.文件句柄管理 2.3.互斥锁 3.注意事项 3.1.禁止复制 3.2.对底层资源使用引用计数法 3.3.复制底部资源(深拷贝)或者转移资源管理权(移动语义) 4.RAII的优势和挑战 5.总…...
矢量图是什么,有哪些格式的文件
矢量图是一种图形设计中常用的图像类型,与我们日常见到的光栅图像(如JPEG、PNG等)有本质的区别。矢量图基于数学方程和几何元素(如点、线、曲线和形状)来表示图像,而不是像光栅图那样通过像素阵列来表示。这…...
Linux 设置快捷命令
以ll命令为例: 在 Linux 系统上,ll 命令通常不是一个独立的程序,而是 ls 命令的一个别名。 这个别名通常在用户的 shell 配置文件中定义,比如 .bashrc 或 .bash_aliases 文件中。 要在 Debian 上启用 ll 命令,你可以按…...
SpringCloudFeign远程调用
文章目录 1. Feign 是什么2. Feign 的使用2.1 引入依赖2.2 写接口2.3 服务调用方2.4 启动测试 3. Feign 日志配置4. Feign 使用优化5. 注意包扫描问题 1. Feign 是什么 Feign 是一个声明式、模板化的 HTTP 客户端,它是由 Netflix 开发并开源的。Feign 极大地简化了…...
Java中List、Set、Map三种集合之间的区别
Java中List、Set、Map三种集合之间的区别 1. List2. Set3. Map 在Java中,List、Set和Map是三种常见的集合类型,它们之间也有一些重要的区别: 1. List List是有序集合,可以存储重复元素。List的实现类常见有ArrayList、LinkedLis…...
SpringMVC之DispatcherServlet组件
目录 一、SpringMVC的核心处理流程二、DispatcherServlet1、init()方法2、doDispatch()方法3、AbstractAnnotationConfigDispatcherServletInitializer类 一、SpringMVC的核心处理流程 请求到达 DispatcherServlet DispatcherServlet 的请求处理: DispatcherServlet…...
抢商家、夺用户、比低价,抖音、快手、小红书“奇招尽出”
随着流量红利逐渐消退,国内电商平台之间互相内卷已成为了行业常态,而无论是在该领域深耕已久的淘宝、京东、拼多多等电商巨头,还是新跨界而来的抖音、快手、小红书等电商新秀都在不断地进行创新,以便为商家提供更好的服务…...
ChatGPT引领的AI面试攻略系列:AI全栈工程师篇
系列文章目录 AI全栈工程师(本文) 文章目录 系列文章目录一、前言二、面试题1. 基础理论与数据处理2. 机器学习3. 深度学习4. 大模型与迁移学习5. 计算机视觉6. 自然语言处理(NLP)7. 多模态学习8. AI生成内容(AIGC&am…...
上位机图像处理和嵌入式模块部署(qmacvisual配置)
【 声明:版权所有,欢迎转载,请勿用于商业用途。 联系信箱:feixiaoxing 163.com】 前面我们谈到了qmacvisual的编译、验证码、用户登录以及流程的编辑,这部分都是基础工作。事实上,除了这些内容之外…...
EXPLAIN PLAN FOR:在Oracle中生成执行计划
目录 案例 解析 Operation类型 在Oracle中,可以使用 EXPLAIN PLAN FOR 命令来生成执行计划,然后通过 SELECT plan_table_output FROM TABLE(DBMS_XPLAN.DISPLAY(PLAN_TABLE))来查看执行计划。需要注意的是,这两个命令需要在同一个窗口下运…...
蓝桥杯 9241.飞机降落
这道题本来作者以为是可以用一些小技巧进行暴力解法的,但是后来试了一下,不能过去全部数据。 下面是对半个的题解: #include<iostream> #include<stdio.h> #include<cstring> #include<cstdlib> #include<cmath…...
数据可视化原理-腾讯-散点图
在做数据分析类的产品功能设计时,经常用到可视化方式,挖掘数据价值,表达数据的内在规律与特征展示给客户。 可是作为一个产品经理,(1)如果不能够掌握各类可视化图形的含义,就不知道哪类数据该用…...
深度学习-Pytorch实现经典AlexNet网络:山高我为峰
深度学习-Pytorch实现经典AlexNet网络之山高我为峰 深度学习中,经典网络引领一波又一波的技术革命,从LetNet到当前最火的GPT所用的Transformer,它们把AI技术不断推向高潮。2012年AlexNet大放异彩,它把深度学习技术引领第一个高峰…...
25考研习题记录
3月 汤家凤《1800》 基础篇 日期高等数学线性代数概率论3.1 P92-93 P212-214 3.4 P10-15 P10-19 极限题62题 P73-74 P170-172 行列式17题 考研竞赛凯哥每日一题 张宇高数30讲页数3.4P74...
上海计算机学会 2023年12月月赛 丙组T4 迷宫(宽度优先搜索)
第四题:T4迷宫 标签:宽度优先搜索题意:给定 n n nx m m m由 # \# #(墙)、 . . .(空地)组成的地图,求从左上角到右下角的最少步数,每次只允许上下左右移动一格࿰…...
【Boost搜索引擎项目】Day1 项目介绍+去标签和数据清洗框架搭建
🌈欢迎来到C项目专栏 🙋🏾♀️作者介绍:前PLA队员 目前是一名普通本科大三的软件工程专业学生 🌏IP坐标:湖北武汉 🍉 目前技术栈:C/C、Linux系统编程、计算机网络、数据结构、Mysq…...
站群服务器需要多大内存
站群服务器的内存需求取决于网站的数量和流量,以及服务器需要运行的应用和服务。RAKsmart小编为您整理发布站群服务器需要多大内存以及站群服务器内存需求的考虑因素。 站群服务器是一种用于托管多个网站的服务器,通常用于搜索引擎优化(SEO)和网络内容管…...
HTB Perfection
Perfection User Namp ┌──(kali㉿kali)-[~/HTB/machine/Perfection] └─$ nmap -A 10.129.226.58 Starting Nmap 7.94SVN ( https://nmap.org ) at 2024-03-03 21:10 EST Nmap scan report for 10....
如何远程连接MySQL数据库?
在现代互联网时代,远程连接MySQL数据库成为了许多开发者和管理员必备的技能。这不仅方便了数据的共享和管理,还可以使多个团队在全球范围内协同工作。本文将介绍如何通过天联组网实现远程连接MySQL数据库,并实现高效的信息远程通信。 天联组网…...
【 HTML 及浏览器 】前端跨页面通信
前端跨页面通信:连接分散界面的纽带 在构建复杂的前端应用时,我们常常需要在不同的页面之间进行数据通信。无论是同源页面还是非同源页面,通信机制都是实现多页面数据同步和交互的关键。本文将探讨各种前端跨页面通信的方法,并提…...
python智能AI技术的中药材店铺管理系统 中药材网上商城系统 46n363df
目录同行可拿货,招校园代理 ,本人源头供货商项目概述核心功能技术栈部署与扩展适用场景项目技术支持获取博主联系方式 源码获取详细视频演示 :同行可合作点击我获取源码->获取博主联系方式->进我个人主页-->同行可拿货,招校园代理 ,本人源头供货商 项目概…...
Pixel 6有锁机保姆级解锁教程:从‘SIM卡不受支持’到完美VoLTE通话(附ADB/Shizuku工具包)
Pixel 6有锁机完全解锁指南:从网络锁到功能优化全攻略 前言 当你从二手市场淘到一台Pixel 6,满心欢喜地插入SIM卡准备使用时,屏幕上却赫然显示"SIM卡不受支持"——这种挫败感我深有体会。作为一款硬件配置出色的设备,Pi…...
凡亿AD最小系统板--元件模型组成介绍
一、课程整体概述原理图库创建是PCB设计流程的第二步核心工作,也是电子设计建模的源头工序。所有电路板上的电子元器件,都需要先在原理图库中绘制对应的电气模型,才能完成后续原理图绘制、封装匹配、PCB布局布线等操作。本系列课程将通过多类…...
3D-LLM:面向可制造性的三维语言模型技术解析
1. 项目概述:当大语言模型开始“看见”三维空间“From Text to Tangible: 3D-LLM Unleashes Language Models into the 3D World”——这个标题不是科幻小说的副标题,而是2024年真实出现在CVPR和ICML顶会workshop上的技术路线宣言。我第一次在arXiv上读到…...
DataRoom开源大屏设计器:零代码打造专业数据可视化大屏的终极指南
DataRoom开源大屏设计器:零代码打造专业数据可视化大屏的终极指南 【免费下载链接】DataRoom 🔥基于SpringBoot、MyBatisPlus、ElementUI、G2Plot、Echarts等技术栈的大屏设计器,具备目录管理、DashBoard设计、预览能力,支持MySQL…...
5分钟搞定专业照片水印:Semi-Utils让你的摄影作品瞬间升级
5分钟搞定专业照片水印:Semi-Utils让你的摄影作品瞬间升级 【免费下载链接】semi-utils 一个批量添加相机机型和拍摄参数的工具,后续「可能」添加其他功能。 项目地址: https://gitcode.com/gh_mirrors/se/semi-utils 还在为照片添加水印而烦恼吗…...
Godot PCK Explorer:可视化浏览与精准定位Godot游戏资源
1. 这不是“解包工具”,而是Godot游戏资产的显微镜 你有没有遇到过这种情况:下载了一个开源Godot游戏,想看看它的UI是怎么做的,动画资源放哪儿,或者想复用某个粒子特效——结果打开文件夹只看到一个几百MB的 game.pc…...
RV1126B开发板驱动多路AHD摄像头:硬件连接、内核驱动与AI应用实战
1. 项目概述:RV1126B开发板与AHD摄像头的融合应用在嵌入式视觉和边缘计算项目中,将传统的模拟高清摄像头接入到高性能的AI计算平台上,是一个既常见又充满挑战的需求。我最近在基于瑞芯微RV1126B芯片的EASY-EAI Nano-TB开发板上,成…...
如果你还在为CAD、SolidWorks的许可发愁,看看这八家
先讲个真事。上个月我一个老同事打电话来,他们公司做非标自动化,四十几个机械工程师,用的主要是SolidWorks和AutoCAD。他说每年买浮动许可的钱快三百万了,结果研发那边还是天天有人排队等许可。他去看了一眼,下午两点半…...
如何3步获取Beyond Compare 5永久授权密钥:开源工具全攻略
如何3步获取Beyond Compare 5永久授权密钥:开源工具全攻略 【免费下载链接】BCompare_Keygen Keygen for BCompare 5 项目地址: https://gitcode.com/gh_mirrors/bc/BCompare_Keygen 还在为Beyond Compare 5的30天试用期到期而烦恼吗?想要免费解锁…...
