0

0

C++ 什么是RAII C++ 资源获取即初始化设计模式详解【理念】

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-02-03 15:23:02

|

512人浏览过

|

来源于php中文网

原创

RAII的本质是构造函数获取资源、析构函数释放资源,要求资源申请必须在构造中完成并确保可用,释放必须且仅在noexcept析构中执行,适用于内存、文件、锁等所有成对资源操作。

c++ 什么是raii c++ 资源获取即初始化设计模式详解【理念】

RAII 的本质是构造函数获取资源、析构函数释放资源

RAII(Resource Acquisition Is Initialization)不是语法特性,而是一种强制绑定资源生命周期与对象生命周期的设计约定。它的核心判断标准只有一条:资源的申请必须发生在构造函数中,且必须确保一旦构造成功,资源就处于可用状态;资源的释放必须且仅发生在析构函数中,且析构函数不得抛出异常。

常见误用是把“带构造/析构的类”等同于 RAII——比如一个类在构造时打开文件但没检查 fopen 返回值,或析构时调用 fclose 却忽略失败可能,这都不算真正 RAII,因为资源状态不可靠。

  • 必须在构造函数内完成资源获取,并做完整错误处理(如抛异常或设置状态位)
  • 析构函数必须是 noexcept,且内部释放逻辑不能失败(如 close() 失败应记录而非抛出)
  • 禁止裸指针管理资源:像 new 出来的内存若由用户手动 delete,就脱离了 RAII 范畴

std::unique_ptr 和 std::shared_ptr 是 RAII 的标准实现

它们不是“支持 RAII”,而是 RAII 在动态内存管理上的直接落地。关键在于:它们的构造函数接管原始指针(或执行 new),析构函数自动调用 delete 或自定义删除器。

区别在于所有权语义:

立即学习C++免费学习笔记(深入)”;

  • std::unique_ptr:独占所有权,移动后原对象为空,不可拷贝
  • std::shared_ptr:共享所有权,引用计数为 0 时才释放资源;注意循环引用会导致资源泄漏
  • 自定义删除器必须是 noexcept,否则破坏 RAII 的析构安全性

示例:std::unique_ptr fp(fopen("log.txt", "w"), &fclose); —— 文件句柄被严格绑定到 fp 对象生命周期。

RAII 不止用于内存,也适用于文件、互斥锁、套接字等所有稀缺资源

只要资源需要成对操作(获取/释放、加锁/解锁、绑定/解绑),就适合用 RAII 封装。C++ 标准库已提供多个现成工具

  • std::lock_guardstd::unique_lock:封装 mutex.lock()/unlock(),避免忘记解锁或异常跳过
  • std::fstream:构造时可打开文件,析构时自动关闭(但注意:不保证刷新缓冲区,显式调用 close() 更稳妥)
  • 自定义 RAII 类常需禁用拷贝(= delete),允许移动(如文件句柄、socket fd)

陷阱:某些系统 API(如 Linux 的 epoll_ctl)没有天然配对操作,需人工抽象出“资源类型”再封装——比如把 epoll_fd 和注册的 fd 一起打包进一个类,析构时统一 epoll_ctl(EPOLL_CTL_DEL)close()

RAII 失效的典型场景:异常未被拦截、跨线程转移、返回局部对象引用

RAII 安全的前提是对象生命周期清晰可控。以下情况会绕过析构函数,导致资源泄漏:

  • 在构造函数中抛异常:若部分资源已分配但后续步骤失败,需在 catch 块中清理(或使用“两阶段构造”,但违背 RAII 简洁性)
  • longjmp 或信号处理函数跳转出作用域:C++ 标准不保证此时析构函数被调用
  • 将 RAII 对象的内部裸资源(如 get() 返回的指针)传递给外部并长期持有,而原对象已销毁
  • 多线程中将 std::unique_ptr 所有权转移到另一线程,但未同步或使用 std::move,导致双重析构或空悬指针

最易被忽略的一点:RAII 解决的是单次作用域内的确定性释放,它不解决资源竞争、全局初始化顺序、或跨模块生命周期协调问题——这些得靠更高层设计(如服务定位器、依赖注入)补足。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
resource是什么文件
resource是什么文件

Resource文件是一种特殊类型的文件,它通常用于存储应用程序或操作系统中的各种资源信息。它们在应用程序开发中起着关键作用,并在跨平台开发和国际化方面提供支持。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

162

2023.12.20

fclose函数的用法
fclose函数的用法

fclose是一个C语言和C++中的标准库函数,用于关闭一个已经打开的文件,是文件操作中非常重要的一个函数,用于将文件流与底层文件系统分离,释放相关的资源。更多关于fclose函数的相关问题,详情请看本专题下面的文章。php中文网欢迎大家前来学习。

334

2023.11.30

线程和进程的区别
线程和进程的区别

线程和进程的区别:线程是进程的一部分,用于实现并发和并行操作,而线程共享进程的资源,通信更方便快捷,切换开销较小。本专题为大家提供线程和进程区别相关的各种文章、以及下载和课程。

568

2023.08.10

Python 多线程与异步编程实战
Python 多线程与异步编程实战

本专题系统讲解 Python 多线程与异步编程的核心概念与实战技巧,包括 threading 模块基础、线程同步机制、GIL 原理、asyncio 异步任务管理、协程与事件循环、任务调度与异常处理。通过实战示例,帮助学习者掌握 如何构建高性能、多任务并发的 Python 应用。

235

2025.12.24

java多线程相关教程合集
java多线程相关教程合集

本专题整合了java多线程相关教程,阅读专题下面的文章了解更多详细内容。

21

2026.01.21

C++多线程相关合集
C++多线程相关合集

本专题整合了C++多线程相关教程,阅读专题下面的的文章了解更多详细内容。

19

2026.01.21

数据库Delete用法
数据库Delete用法

数据库Delete用法:1、删除单条记录;2、删除多条记录;3、删除所有记录;4、删除特定条件的记录。更多关于数据库Delete的内容,大家可以访问下面的文章。

279

2023.11.13

drop和delete的区别
drop和delete的区别

drop和delete的区别:1、功能与用途;2、操作对象;3、可逆性;4、空间释放;5、执行速度与效率;6、与其他命令的交互;7、影响的持久性;8、语法和执行;9、触发器与约束;10、事务处理。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

215

2023.12.29

全国统一发票查询平台入口合集
全国统一发票查询平台入口合集

本专题整合了全国统一发票查询入口地址合集,阅读专题下面的文章了解更多详细入口。

4

2026.02.03

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 8.4万人学习

Git 教程
Git 教程

共21课时 | 3.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号