0

0

C++中的RAII是什么意思?(资源获取即初始化)

尼克

尼克

发布时间:2026-01-20 16:28:33

|

234人浏览过

|

来源于php中文网

原创

RAII是C++管理资源的底层契约,要求资源在构造时获取、析构时无条件释放,依赖栈对象生命周期自动保证,不靠手动调用或try/finally。

c++中的raii是什么意思?(资源获取即初始化)

RAII 不是语法糖,也不是可选技巧,而是 C++ 管理资源(内存、文件、锁、句柄等)的底层契约。它要求:资源必须在对象构造时获取,在析构时无条件释放——靠对象生命周期自动保证,不依赖程序员手动调用 close()delete

为什么不用 try/finally 或手动 free?

C++ 没有 finally,异常可能在任意位置抛出;手动管理极易漏掉释放点,尤其在多分支或早期返回场景下。RAII 把“获取-释放”绑定到对象生存期,由编译器保证析构执行(除非 std::terminate 被触发)。

常见错误现象:
• 函数中途 return 导致 fclose(fp) 未执行
• 异常传播跳过 delete ptr
• 多线程中忘记 pthread_mutex_unlock()

  • RAII 对象必须是栈分配(或智能指针托管),堆上裸 new 的对象无法自动析构
  • 析构函数不能抛异常(否则栈展开时二次崩溃),需用 noexcept 显式标注
  • 移动语义下,资源所有权应转移而非复制(如 std::unique_ptrmove 构造)

典型 RAII 类怎么写?

以封装文件描述符为例,关键在于:构造函数负责 open(),析构函数负责 close(),且禁止拷贝(避免双重关闭):

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

class FileDescriptor {
    int fd_;
public:
    explicit FileDescriptor(const char* path) : fd_(open(path, O_RDONLY)) {
        if (fd_ == -1) throw std::runtime_error("open failed");
    }
    ~FileDescriptor() noexcept { 
        if (fd_ != -1) close(fd_); 
    }
    FileDescriptor(const FileDescriptor&) = delete;
    FileDescriptor& operator=(const FileDescriptor&) = delete;
    FileDescriptor(FileDescriptor&& other) noexcept : fd_(other.fd_) { 
        other.fd_ = -1; 
    }
    FileDescriptor& operator=(FileDescriptor&& other) noexcept {
        if (this != &other) {
            if (fd_ != -1) close(fd_);
            fd_ = other.fd_;
            other.fd_ = -1;
        }
        return *this;
    }
};

使用时:FileDescriptor f("/etc/passwd"); —— 出作用域自动关 fd,无需关心 return 或异常路径。

甲骨文AI协同平台
甲骨文AI协同平台

专门用于甲骨文研究的革命性平台

下载

标准库里哪些是 RAII?

几乎所有带“自动管理”语义的类都是 RAII 实现:

  • std::vector:构造时分配内存,析构时 delete[]
  • std::lock_guard:构造时加锁,析构时解锁(即使临界区 throw 也安全)
  • std::unique_ptr:构造时接管裸指针,析构时 delete
  • std::fstream:构造时可打开文件,析构时自动 close()

注意:std::shared_ptr 是 RAII,但它的析构只减少引用计数;真正释放资源发生在最后一个引用消失时——这仍是 RAII,只是延迟了释放时机。

容易被忽略的坑

RAII 的可靠性高度依赖正确实现。几个隐蔽陷阱:

  • 析构函数里调用虚函数:此时虚表已部分销毁,行为未定义
  • 在构造函数中抛异常:对象未完全构造,析构函数不会被调用(但已构造的成员会析构)
  • 把 RAII 对象存进容器后又用 std::move:原对象状态必须置为“无效但可析构”(如 fd 设为 -1)
  • 跨 DLL 边界传递 RAII 对象:若两边 STL 版本/ABI 不一致,析构可能崩溃

最常被低估的一点:RAII 解决的是“何时释放”,不是“是否该释放”。如果资源本身不该被自动释放(比如全局 OpenGL 上下文),强行套 RAII 反而引入错误语义。

相关专题

更多
fclose函数的用法
fclose函数的用法

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

328

2023.11.30

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

392

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

572

2023.08.10

堆和栈的区别
堆和栈的区别

堆和栈的区别:1、内存分配方式不同;2、大小不同;3、数据访问方式不同;4、数据的生命周期。本专题为大家提供堆和栈的区别的相关的文章、下载、课程内容,供大家免费下载体验。

392

2023.07.18

堆和栈区别
堆和栈区别

堆(Heap)和栈(Stack)是计算机中两种常见的内存分配机制。它们在内存管理的方式、分配方式以及使用场景上有很大的区别。本文将详细介绍堆和栈的特点、区别以及各自的使用场景。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

572

2023.08.10

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

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

481

2023.08.10

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

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

143

2025.12.24

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

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

269

2023.11.13

Java JVM 原理与性能调优实战
Java JVM 原理与性能调优实战

本专题系统讲解 Java 虚拟机(JVM)的核心工作原理与性能调优方法,包括 JVM 内存结构、对象创建与回收流程、垃圾回收器(Serial、CMS、G1、ZGC)对比分析、常见内存泄漏与性能瓶颈排查,以及 JVM 参数调优与监控工具(jstat、jmap、jvisualvm)的实战使用。通过真实案例,帮助学习者掌握 Java 应用在生产环境中的性能分析与优化能力。

6

2026.01.20

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1.0万人学习

进程与SOCKET
进程与SOCKET

共6课时 | 0.3万人学习

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

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