智能指针是C++11引入的自动内存管理机制,核心在于正确选用unique_ptr(独占、零开销)或shared_ptr(共享、带引用计数),避免裸指针混用、循环引用等误用。

智能指针是 C++11 引入的核心机制,用来自动管理动态分配的内存,避免手动 new/delete 带来的泄漏、重复释放和悬空指针问题。关键不是“用不用”,而是“怎么用对”——shared_ptr 和 unique_ptr 解决不同场景,选错反而引入新问题。
unique_ptr:独占所有权,轻量高效
unique_ptr 表示资源只能被一个指针拥有,不可复制,但可移动。它几乎零开销(不带引用计数),适合绝大多数需要堆内存但不需要共享的场景,比如容器元素、函数返回临时资源、RAII 封装等。
- 创建用
std::make_unique,别直接 new —— 安全且简洁:() auto p = std::make_unique(42); - 不能赋值或拷贝:
auto q = p; // ❌ 编译失败auto r = std::move(p); // ✅ 转移所有权,p 变为空 - 支持自定义删除器(比如关闭文件、释放 C 风格资源):
auto fp = std::unique_ptr(fopen("a.txt", "r"), &fclose);
shared_ptr:共享所有权,带引用计数
shared_ptr 允许多个指针共同拥有同一块资源,内部维护引用计数。当最后一个 shared_ptr 被销毁或重置时,资源才被释放。适用于真正需要共享访问的场景,比如对象图、观察者模式、缓存等。
- 优先用
std::make_shared创建 —— 一次分配对象+控制块,比() new+ 构造更高效:auto sp = std::make_shared<:string>("hello"); - 可拷贝、可赋值,引用计数自动增减:
auto sp2 = sp; // 计数+1sp.reset(); // 计数-1,若为0则释放资源 - 注意循环引用:两个
shared_ptr互相持有会导致计数永不归零。破环用weak_ptr(只观察不参与计数):std::weak_ptrparent_ref;
常见误用与避坑点
智能指针不是万能胶,用错会埋雷:
立即学习“C++免费学习笔记(深入)”;
-
别混用裸指针和智能指针管理同一块内存 —— 比如用
new分配后又用shared_ptr接管,再手动delete,必崩溃。 -
别把
this直接传给shared_ptr—— 会造成双重管理。要用enable_shared_from_this:class A : public std::enable_shared_from_this { ... };auto self = shared_from_this(); // ✅ 安全获取 shared_ptr -
数组要显式指定删除器或用
shared_ptr(C++17 起支持):auto arr = std::make_shared(100); // ✅ auto raw = std::shared_ptr(new int[100], [](int* p) { delete[] p; }); // ❌ 过时写法,尽量避免
怎么选?一个简单判断标准
问自己一个问题:这块内存,是否**必须**由多个所有者共同决定生命周期?
- 否 → 用
unique_ptr(默认选择,更安全、更快、更清晰) - 是 → 再确认是否真无法用其他方式解耦(比如消息传递、回调、
weak_ptr观察);如果确实需要共享 → 用shared_ptr,并警惕循环引用
基本上就这些。不复杂,但容易忽略细节。









