C++ 独占指针被销毁后,堆也会自己销毁

独占指针被销毁后,堆也会自己销毁吗

1、结论:是的,std::unique_ptr 析构时会自动释放它托管的堆内存

unique_ptr独占式 RAII 智能指针 ,生命周期和托管堆对象强绑定:只要 unique_ptr 本身被销毁(出作用域、容器销毁、赋值覆盖等),它内部会自动执行 delete 释放堆上的对象。

对应你代码里的场景

cpp

运行

复制代码
// 1. 创建堆对象,task 独占持有
auto task = std::make_unique<AudioTask>();

// 2. 所有权转移到队列容器
audio_encode_queue_.push_back(std::move(task));
// 此时局部task是空(nullptr),不再管AudioTask

// 3. 编码线程取出,所有权给到局部task_ptr
auto task_ptr = std::move(audio_encode_queue_.front());
audio_encode_queue_.pop_front();

// 4. task_ptr 代码块结束、变量销毁 → 自动 delete AudioTask
// AudioTask内部的 std::vector<int16_t> pcm 会跟着自身析构,释放pcm的堆缓冲区

2、分场景细化销毁行为

场景 A:局部 unique_ptr 正常出栈

cpp

运行

复制代码
void test()
{
    auto ptr = std::make_unique<int>(100); // 堆int创建,ptr持有
} // 函数结束,ptr栈变量销毁 → 自动 delete 堆int
场景 B:容器存 unique_ptr,容器销毁

cpp

运行

复制代码
std::vector<std::unique_ptr<AudioTask>> queue;
queue.emplace_back(std::make_unique<AudioTask>());
// 程序退出/queue被销毁时,vector会逐个析构内部所有unique_ptr
// 每一个都会delete对应的AudioTask堆实例
场景 C:std::move 转移后,旧指针自动置空、不会二次释放

cpp

运行

复制代码
auto a = std::make_unique<int>(5);
auto b = std::move(a);
// a 变成空指针,a销毁时什么都不做;b持有堆内存
// b销毁时才执行delete

核心:同一堆内存永远只有一个 unique_ptr 拥有释放权,绝不会重复 delete

场景 D:手动 reset 主动释放

cpp

运行

复制代码
auto ptr = std::make_unique<AudioTask>();
ptr.reset(); // 立刻释放堆对象,ptr变为空

3、嵌套内存释放(你 AudioTask 的结构)

cpp

运行

复制代码
struct AudioTask {
    AudioTaskType type;
    uint64_t timestamp;
    std::vector<int16_t> pcm; // vector自身在AudioTask堆内存里
};
  1. unique_ptr<AudioTask> 析构 → delete AudioTask
  2. AudioTask 整体内存释放前,内部成员 pcm 先执行自身析构
  3. std::vector 析构会释放它内部存储 PCM 采样点的堆数组整条内存链全自动释放,无内存泄漏

4、对比裸指针(反例,方便理解差距)

cpp

运行

复制代码
// 裸指针不会自动释放
AudioTask* raw = new AudioTask();
// 函数结束raw栈变量销毁,堆AudioTask永远留在内存 → 内存泄漏
delete raw; // 必须手动写,忘记就漏

极简一句话记忆

unique_ptr = 自带自动 delete 的裸指针;持有者消失,堆内存同步销毁;全程独占,安全无重复释放、无泄漏。

相关推荐
郝学胜_神的一滴13 小时前
CMake 034:生成器表达式:解耦构建时序、精简分支逻辑的终极利器
c++·cmake
见过夏天1 天前
C++ 基础入门完全指南
c++
用户805533698033 天前
不止三件套:QObject 属性系统全关键字与运行时反射!
c++·qt
BadBadBad__AK3 天前
线段树维护区间 k 次方和
c++·数学·算法·stl
卷无止境4 天前
Eigen 库如何借助 OpenMP 加速计算
c++·后端
卷无止境4 天前
OpenMPI、MPICH 与 OpenMP:关系、核心概念与架构全解
c++·后端
郝学胜_神的一滴5 天前
CMake 30:循环语法全解|foreach_while双循环精讲、迭代技巧与实战避坑指南
c++·cmake
卷无止境7 天前
C++ 的Eigen 库全解析
c++
卷无止境7 天前
现代 C++特性大盘点:一门脱胎换骨的老语言
c++·后端
郝学胜_神的一滴7 天前
CMake 27:缓存变量的特性、语法、类型与实操全解
c++·cmake