0

0

c++如何理解raii资源管理_c++ 构造函数获取与析构函数释放【教程】

冰火之心

冰火之心

发布时间:2026-01-02 09:59:02

|

176人浏览过

|

来源于php中文网

原创

RAII的本质是“作用域绑定生命周期”,即资源生命周期严格绑定到栈对象生存期;构造函数不应抛异常,析构函数必须noexcept,且应避免手动管理资源而优先使用标准RAII工具。

c++如何理解raii资源管理_c++ 构造函数获取与析构函数释放【教程】

RAII 的本质不是“构造获取 + 析构释放”,而是“作用域绑定生命周期”

RAII(Resource Acquisition Is Initialization)常被误解为“在构造函数里 new,在析构函数里 delete”。这不对——真正关键的是:**资源的生命周期必须严格绑定到某个对象的生存期上**。构造函数是否真的“获取”资源、析构函数是否真的“释放”,取决于资源类型和使用意图。比如 std::lock_guard 构造时加锁、析构时解锁;std::unique_ptr 构造时接管裸指针、析构时调用 delete;但 std::string 构造时分配内存,析构时释放,你甚至不感知它用了堆——它仍是 RAII。

构造函数里不能抛异常,否则析构函数不会被调用

这是 RAII 最容易翻车的地方。如果构造函数中途抛出异常,对象未完全构造成功,C++ 标准规定:**该对象的析构函数绝不会执行**。此时若你在构造中已手动 new 了资源(比如文件句柄、内存、socket),就彻底泄漏了。

  • 永远不要在构造函数里做可能失败的资源获取操作(如打开文件、连接网络、malloc 失败)
  • 若必须处理可能失败的初始化,改用工厂函数 + std::optional 或返回 std::expected(C++23)
  • 更稳妥的做法是把资源获取延迟到一个独立的 init() 成员函数,并由用户显式调用(但这就脱离 RAII 了)
class BadRAII {
    FILE* fp;
public:
    BadRAII(const char* name) {
        fp = std::fopen(name, "r");
        if (!fp) throw std::runtime_error("open failed"); // ❌ 析构不执行,fp 泄漏
    }
    ~BadRAII() { if (fp) std::fclose(fp); } // 永远不会被调用
};

析构函数必须是 noexcept,否则栈展开会直接终止程序

C++ 要求析构函数默认为 noexcept(除非显式声明 noexcept(false))。如果析构函数意外抛出异常(比如 fclose() 失败且你写了 throw),而此时程序已在栈展开过程中(例如另一个异常正在传播),std::terminate() 会被立即调用——进程静默退出,无日志、无调试线索。

  • 所有 RAII 类的析构函数应避免任何可能抛异常的操作
  • 系统调用如 close()fclose()pthread_mutex_destroy() 等失败时只设 errno,不应转为异常
  • 若需报告析构期错误(极少见),应记录日志或写入全局状态,而非抛出
class SafeFile {
    FILE* fp;
public:
    SafeFile(const char* name) : fp(std::fopen(name, "w")) {}
    ~SafeFile() noexcept { // ✅ 显式标记
        if (fp) std::fclose(fp);
    }
};

别用裸指针 + 手动 new/delete 模拟 RAII

有人写这样的“伪 RAII”:

arXiv Xplorer
arXiv Xplorer

ArXiv 语义搜索引擎,帮您快速轻松的查找,保存和下载arXiv文章。

下载

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

class ManualPtr {
    int* p;
public:
    ManualPtr() : p(new int(42)) {}
    ~ManualPtr() { delete p; }
    int* get() { return p; }
};

问题在于:它不满足 RAII 的三大支柱——没有移动语义(C++11 后必须支持)、没有禁止拷贝(导致双重释放)、无法组合(比如放进 std::vector 就崩)。正确做法是直接用标准工具

  • 动态内存 → std::unique_ptrstd::shared_ptr
  • 文件描述符 → std::filesystem::path 配合 RAII 封装类,或 folly::File 等第三方
  • 互斥锁 → std::lock_guard / std::unique_lock
  • 自定义资源 → 继承 std::unique_ptr 的删除器,或写最小化封装类(含移动构造/赋值,禁拷贝)

自己手写 RAII 类不是不行,但得完整实现移动语义、noexcept 析构、异常安全构造——多数时候,复用标准库更可靠。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

312

2023.08.02

resource是什么文件
resource是什么文件

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

142

2023.12.20

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

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

322

2023.11.30

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

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

371

2023.07.18

堆和栈区别
堆和栈区别

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

563

2023.08.10

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

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

371

2023.07.18

堆和栈区别
堆和栈区别

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

563

2023.08.10

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

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

266

2023.11.13

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

74

2025.12.31

热门下载

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

精品课程

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

共28课时 | 4万人学习

PostgreSQL 教程
PostgreSQL 教程

共48课时 | 6.4万人学习

Git 教程
Git 教程

共21课时 | 2.3万人学习

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

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