0

0

如何在C++中优雅地管理内存生命周期?(RAII机制)

冰火之心

冰火之心

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

|

245人浏览过

|

来源于php中文网

原创

raii是c++中不可替代的资源管理契约,通过构造函数获取、析构函数释放资源,将生命周期绑定作用域;裸指针易致泄漏或悬空,应优先使用std::make_unique/std::make_shared,避免手动new/delete及循环引用。

如何在c++中优雅地管理内存生命周期?(raii机制)

RAII不是语法糖,是C++里唯一靠谱的资源管理契约

RAII(Resource Acquisition Is Initialization)不是可选技巧,而是C++中内存不泄漏、对象不悬空的前提。它靠构造函数获取资源、析构函数释放资源,把生命周期绑定到作用域上——只要对象能被正确销毁,资源就一定被释放。

关键点在于:你不能绕过它去“手动管理”,否则delete漏掉一次,或者异常中途跳出作用域,delete就永远没机会执行。

  • 所有裸指针(int*MyClass*)都不该在栈上长期持有,除非你明确知道它指向的是静态/全局/已托管内存
  • newdeletenew[]delete[]必须严格一一对应;混用会触发未定义行为,常见表现为double free or corruption
  • 函数返回裸指针?基本等于把资源管理责任甩给调用方——而调用方大概率不知道该不该delete

std::unique_ptr 是最常用也最容易写错的智能指针

std::unique_ptr本质是“独占语义”的栈上对象,它自己不分配堆内存,只管理别人分配的堆内存。它的析构函数自动调用delete(或delete[]),但前提是构造时传入了合法指针。

常见错误现象:std::unique_ptr<int> p(new int(42));</int>看起来没问题,但如果new抛异常(比如内存耗尽),p根本不会被构造,new分配的内存就泄露了——这正是std::make_unique存在的理由。

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

Rezi.ai
Rezi.ai

一个使用 AI 自动化创建简历平台

下载
  • 永远优先用std::make_unique<t>(...)</t>,而不是std::unique_ptr<t>(new T(...))</t>
  • 想传递所有权?用std::move(p),之后p变成nullptr;直接赋值或拷贝会编译报错(copy constructor is deleted
  • 数组要用std::unique_ptr<int></int>,且必须用std::make_unique<int>(n)</int>,不能用new int[n]配普通unique_ptr<int></int>

std::shared_ptr 的循环引用不是玄学,是引用计数模型的必然结果

std::shared_ptr靠引用计数管理生命周期,当计数归零才释放资源。但它无法识别两个shared_ptr互相持有对方——比如A对象里存着指向B的std::shared_ptr<b></b>,B里又存着指向A的std::shared_ptr<a></a>,那么即使A和B都离开作用域,它们的引用计数也永远不会降到0。

典型表现:程序退出前内存没释放,Valgrind报告“still reachable”块,shared_ptr析构函数压根没被调用。

  • 类内部需要反向引用外部对象时,用std::weak_ptr替代std::shared_ptr,访问前调用lock()检查是否还有效
  • std::shared_ptr的控制块(control block)额外分配内存,比unique_ptr有轻微开销;频繁拷贝shared_ptr本身成本低,但大量创建/销毁会触发控制块分配
  • 不要用shared_ptr管理栈对象或全局对象(比如std::shared_ptr<int>(&x)</int>),析构时会delete栈地址,直接崩溃

自定义类型必须显式实现移动语义才能安全参与RAII

如果你写了带std::unique_ptr成员的类,却没声明移动构造函数和移动赋值运算符,编译器生成的默认版本会尝试拷贝unique_ptr——这会导致编译失败(因为unique_ptr不可拷贝)。更糟的是,如果成员是裸指针,而你没写移动操作,编译器会生成浅拷贝,两个对象指向同一块内存,析构时double free

一个没写移动语义的类,放进std::vector扩容时可能崩溃,传参时可能意外复制,返回局部对象时可能被强制拷贝(C++11前)或静默失败(C++11后)。

  • 只要类里有资源(裸指针、文件句柄、socket等),就必须考虑移动语义:用= default让编译器生成,或手写并确保转移后源对象处于有效但未定义状态
  • 禁用拷贝:把拷贝构造函数和拷贝赋值设为= delete,避免误用
  • 析构函数里别抛异常——RAII的清理阶段一旦抛异常,程序直接std::terminate

RAII的边界其实很窄:它只管“对象生命周期结束时该做什么”,不管“什么时候该结束”。决定生命周期的,是你把对象放在哪——栈上?容器里?还是shared_ptr堆上?选错位置,再严谨的RAII也救不了。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

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

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

179

2023.12.20

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1564

2023.10.24

Go语言中的运算符有哪些
Go语言中的运算符有哪些

Go语言中的运算符有:1、加法运算符;2、减法运算符;3、乘法运算符;4、除法运算符;5、取余运算符;6、比较运算符;7、位运算符;8、按位与运算符;9、按位或运算符;10、按位异或运算符等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

241

2024.02.23

php三元运算符用法
php三元运算符用法

本专题整合了php三元运算符相关教程,阅读专题下面的文章了解更多详细内容。

128

2025.10.17

string转int
string转int

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

990

2023.08.02

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

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

607

2024.08.29

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

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

314

2025.08.29

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

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

235

2025.08.29

Go高并发任务调度与Goroutine池化实践
Go高并发任务调度与Goroutine池化实践

本专题围绕 Go 语言在高并发任务处理场景中的实践展开,系统讲解 Goroutine 调度模型、Channel 通信机制以及并发控制策略。内容包括任务队列设计、Goroutine 池化管理、资源限制控制以及并发任务的性能优化方法。通过实际案例演示,帮助开发者构建稳定高效的 Go 并发任务处理系统,提高系统在高负载环境下的处理能力与稳定性。

4

2026.03.10

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
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号