智能指针是c++动态内存管理的底线:unique_ptr用于独占所有权,shared_ptr用于共享所有权并带引用计数,weak_ptr破除循环引用;均基于raii自动管理生命周期,避免内存泄漏、野指针和双重释放。

std::unique_ptr、std::shared_ptr、std::weak_ptr 不是“可选技能”,而是 C++ 动态内存管理的底线——不用它们,几乎必然掉进内存泄漏、野指针或双重释放的坑里。
new/delete 一出错就崩溃,智能指针怎么救?
手动 new 后忘了 delete,或异常中途跳出导致清理代码跳过,是 C++ 最高频的崩溃源头。比如:Divide 抛异常时,array1 和 array2 的 delete[] 根本不会执行。
- 智能指针把资源绑定到对象生命周期上:构造时拿内存,析构时自动还——哪怕函数因异常提前退出,栈上智能指针照样被销毁
- 它不是“帮你写 delete”,而是让
delete的时机完全脱离程序员的手动控制,由 RAII 机制兜底 - 别再写
try/catch套两层去delete,那只是补漏;用std::unique_ptr<int></int>一行替代int* p = new int[10],问题直接消失
该用 unique_ptr 还是 shared_ptr?看所有权是否唯一
std::unique_ptr 是默认首选;只有明确需要“多个地方同时持有同一对象”时,才考虑 std::shared_ptr。
-
std::unique_ptr:零开销、不可拷贝、支持移动;适合工厂函数返回、类成员封装、临时资源管理(如文件句柄) -
std::shared_ptr:带引用计数,线程安全但有轻微性能成本;只在真正共享场景用,比如观察者列表、缓存项、跨线程传递对象 - 错误示范:
std::shared_ptr存储局部对象地址(std::shared_ptr<int>(&x)</int>)→ 析构时试图delete &x,未定义行为 - 正确习惯:永远优先用
std::make_unique或std::make_shared,避免裸new
shared_ptr 循环引用不崩溃,但对象永不释放
这不是 crash,而是静默泄漏:两个 std::shared_ptr 互相持有,引用计数永远 ≥1,析构函数根本不会调用。
立即学习“C++免费学习笔记(深入)”;
- 典型场景:
TreeNode中parent和children都用std::shared_ptr;Subject和Observer相互保存对方的std::shared_ptr - 破局关键:把其中一方改成
std::weak_ptr(比如parent_成员),它不增加引用计数,也不会阻止析构 - 访问前必须调用
.lock():返回std::shared_ptr或空指针,不能直接解引用std::weak_ptr - 别指望编译器报错——这种泄漏在运行时完全无症状,直到内存耗尽或程序卡顿
ROS/多线程里用错智能指针,比内存泄漏更危险
在回调函数、异步任务中,智能指针生命周期和对象实际存活时间错位,会直接触发 Use-After-Free。
- 常见误操作:在函数内创建
std::shared_ptr<pointcloudsubscriber></pointcloudsubscriber>,却只把它的subscriber_(一个ros::Subscriber句柄)存进 map;函数一结束,PointCloudSubscriber对象就被析构,句柄变悬空 - 本质是混淆了“句柄”和“拥有者”:ROS 的
Subscriber必须由其所属对象长期存活才能有效,不能靠句柄续命 - 解决方式:把整个
std::shared_ptr存进容器(如std::map<:string std::shared_ptr>></:string>),确保对象生命周期覆盖所有回调执行期 - 注意:
std::shared_ptr的引用计数操作本身是原子的,但对象内部状态仍需额外同步(如用std::mutex)
智能指针不是语法糖,它是把“谁负责释放”这个模糊问题,变成编译器能验证、运行时能保障的确定性规则。最危险的不是不会用,而是以为自己用了——比如裸 new + shared_ptr 包装、或 weak_ptr 忘了 lock 就解引用。这些点,往往要等压测几天后才暴露。










