0

0

C++ 内存泄漏是什么?(如何利用 RAII 机制彻底避免)

冰火之心

冰火之心

发布时间:2026-02-25 03:51:19

|

926人浏览过

|

来源于php中文网

原创

内存泄漏指未释放动态分配的内存或资源,导致堆内存持续增长;raii通过将资源绑定到对象生命周期来预防,但对全局对象、线程局部存储、dlopen插件及longjmp等场景无效。

c++ 内存泄漏是什么?(如何利用 raii 机制彻底避免)

内存泄漏在 C++ 里到底意味着什么

它不是程序崩溃,而是你 new 了内存却没 delete,或者 malloc 了却没 free,导致进程的堆内存持续增长、最终耗尽。更隐蔽的是:对象构造时申请资源(比如文件句柄、socket),但析构函数没释放——这也算内存泄漏的“远亲”。RAII 不是银弹,但它能让你在 95% 的日常场景里,根本写不出泄漏代码。

RAII 的核心不是“用智能指针”,而是“资源即对象”

关键在于:把资源生命周期绑定到栈对象的构造/析构上。不是所有资源都适合用 std::shared_ptr,也不是所有对象都该被“智能指针化”。真正起作用的是——

  • 资源获取即初始化:构造函数里申请,失败就抛异常,不留下半成品对象
  • 资源释放即销毁:析构函数里无条件释放,哪怕提前 return 或 throw 也保证执行
  • 禁止裸指针跨作用域传递:一旦出现 new 后赋给非 RAII 对象(比如普通 int* 成员),RAII 就失效

例如打开文件:

class FileGuard {
    int fd_;
public:
    FileGuard(const char* path) : fd_(open(path, O_RDONLY)) {
        if (fd_ == -1) throw std::runtime_error("open failed");
    }
    ~FileGuard() { if (fd_ != -1) close(fd_); }
    // 禁用拷贝,允许移动(可选)
    FileGuard(const FileGuard&) = delete;
    FileGuard& operator=(const FileGuard&) = delete;
};

std::unique_ptr 和 std::shared_ptr 哪个更容易踩坑

std::unique_ptr 安全边界清晰:独占所有权、不可拷贝、移动后原指针变空。但常见错误是——

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

LAIKA
LAIKA

LAIKA 是一个创意伙伴,您可以训练它像您(或您想要的任何人)一样写作。

下载
  • get() 拿到裸指针后存到别处(比如传给 C API 并长期持有),忘了自己仍是所有者
  • 循环引用:两个 std::shared_ptr 互相持有时,引用计数永远不为 0,导致泄漏
  • 自定义删除器写错:比如 std::shared_ptr<int>(new int, [](int* p){ delete p; })</int> 看似正常,但如果传入的是数组,delete 就 UB

真要管理动态数组,请用 std::unique_ptr<int></int>,它会自动调用 delete[]

哪些地方 RAII 压根不起作用

RAII 依赖对象生命周期可控。以下场景它管不了——

  • 全局或静态对象的构造函数里分配资源:如果构造失败,C++ 标准不保证其析构会被调用
  • 线程局部存储(thread_local)对象,若线程异常终止,析构可能被跳过
  • 通过 dlopen 加载的插件中分配的资源,主程序卸载时无法触发其 RAII 对象析构
  • longjmp 跳出作用域:栈展开被绕过,析构函数不执行(C++ 中应避免 setjmp/longjmp

这些地方得靠人工检查 + 工具辅助(比如 ASan、Valgrind),RAII 只负责它能管的那一块。

最难防的泄漏,往往藏在“看起来很安全”的封装里:比如一个类内部用 std::vector 存数据,但忘了重写移动构造函数,导致浅拷贝后原始对象析构时清空了本该移交的数据——这种 bug 不报错,只悄悄泄漏。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
string转int
string转int

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

850

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

583

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

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

294

2025.08.29

C++中int的含义
C++中int的含义

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

210

2025.08.29

string转int
string转int

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

850

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

583

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

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

294

2025.08.29

C++中int的含义
C++中int的含义

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

210

2025.08.29

Golang 生态工具与框架:扩展开发能力
Golang 生态工具与框架:扩展开发能力

《Golang 生态工具与框架》系统梳理 Go 语言在实际工程中的主流工具链与框架选型思路,涵盖 Web 框架、RPC 通信、依赖管理、测试工具、代码生成与项目结构设计等内容。通过真实项目场景解析不同工具的适用边界与组合方式,帮助开发者构建高效、可维护的 Go 工程体系,并提升团队协作与交付效率。

1

2026.02.24

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

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

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