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 及浏览器 】前端跨页面通信
前端跨页面通信:连接分散界面的纽带 在构建复杂的前端应用时,我们常常需要在不同的页面之间进行数据通信。无论是同源页面还是非同源页面,通信机制都是实现多页面数据同步和交互的关键。本文将探讨各种前端跨页面通信的方法,并提…...
GLM-4.1V-9B-Base部署案例:中小企业零基础部署视觉AI助手教程
GLM-4.1V-9B-Base部署案例:中小企业零基础部署视觉AI助手教程 1. 为什么选择GLM-4.1V-9B-Base 对于中小企业来说,部署一个视觉AI助手可以解决很多实际问题:产品图片自动分类、客户上传图片内容识别、社交媒体图片分析等。GLM-4.1V-9B-Base是…...
OpenClaw备份自动化:用SecGPT-14B识别关键数据并同步加密
OpenClaw备份自动化:用SecGPT-14B识别关键数据并同步加密 1. 为什么需要智能备份系统 作为一个长期在本地开发项目的程序员,我经历过太多次"误删文件后追悔莫及"的时刻。传统的定时全量备份虽然简单,但存在三个致命问题ÿ…...
OpenClaw隐私保护方案:Qwen3-14b_int4_awq本地化数据处理优势
OpenClaw隐私保护方案:Qwen3-14b_int4_awq本地化数据处理优势 1. 为什么我们需要关注AI自动化中的隐私问题 去年我帮朋友的公司评估一个自动化方案时,遇到一个典型场景:他们需要处理大量客户合同,但担心使用云端AI服务会导致敏感…...
揭秘novel-downloader:从零打造你的专属小说下载器实战指南
揭秘novel-downloader:从零打造你的专属小说下载器实战指南 【免费下载链接】novel-downloader 一个可扩展的通用型小说下载器。 项目地址: https://gitcode.com/gh_mirrors/no/novel-downloader 在数字阅读时代,你是否曾遇到过心爱的小说突然消失…...
Pandas 操作指南(三):数据清洗与预处理
数据能够进入 DataFrame,并不意味着它已经适合直接分析。在实际工作中,原始数据常常存在缺失、重复、格式混乱、类型不当等问题。若不先处理这些问题,后续统计结果就可能失真,筛选逻辑也可能出错。因此,数据清洗&#…...
实战指南:Spring Boot集成Google OAuth 2.0实现免密登录与用户信息同步
1. 为什么需要Google OAuth 2.0登录 在开发面向海外用户的Web应用时,用户注册和登录流程的便捷性直接影响转化率。传统邮箱注册需要用户完成"填写邮箱-接收验证码-设置密码"的繁琐流程,而Google OAuth 2.0登录可以让用户一键完成身份验证。实…...
网站 SEO 检测报告如何与网站分析数据进行对比分析_网站 SEO 检测报告中的页面结构分析有什么用
网站 SEO 检测报告如何与网站分析数据进行对比分析 在当今的互联网时代,网站的成功与否往往取决于其在搜索引擎上的排名。因此,网站 SEO(搜索引擎优化)检测报告和网站分析数据的对比分析显得尤为重要。通过对比分析,可…...
OpenClaw+Qwen3-14b_int4_awq:电商商品描述自动生成器
OpenClawQwen3-14b_int4_awq:电商商品描述自动生成器 1. 为什么需要自动化商品描述生成 作为一名电商运营人员,我每天都要处理大量商品上架工作。最让我头疼的就是为不同语言市场的同一款产品撰写多版本描述——不仅耗时耗力,还经常出现风格…...
2026年最新盘点:全球TOP5高尔夫模拟系统公司,谁将引领行业新标准?
随着科技与体育的深度融合,室内高尔夫模拟系统已成为高尔夫爱好者、专业球员乃至商业场馆不可或缺的装备。它不仅打破了传统高尔夫运动对天气、场地和时间的严苛限制,更通过精准的数据分析,为技术提升提供了科学依据。面对市场上琳琅满目的品…...
OpenClaw的5个国内实用Skill:助力高效办公的智能引擎
在AI智能体日益普及的2026年,OpenClaw作为国内领先的AI执行框架,其核心价值不仅在于底层技术,更在于其丰富的技能生态。尤其在中文办公场景中,经过本土化优化的技能已成为提升效率的关键。本文将当前国内最实用的5个OpenClaw技能&…...
