千问AI教你C++智能指针使用技巧
时间:2026-03-05 15:45:47 310浏览 收藏
本文深入剖析了C++智能指针(std::shared_ptr、std::unique_ptr、std::weak_ptr)在实际开发中极易被忽视却可能导致崩溃、内存泄漏或性能劣化的关键细节:从避免用nullptr直接构造shared_ptr引发的控制块异常,到unique_ptr移动后裸指针立即失效的野指针陷阱;从weak_ptr.lock()必须即时判空的安全使用范式,到自定义deleter如何悄然让shared_ptr体积翻倍并拖累缓存与拷贝性能——每一条都是踩过坑的经验之谈,帮你避开教科书不讲、但线上故障频发的“高危操作”。

std::shared_ptr 初始化空指针时别直接传 nullptr
传 nullptr 给 std::shared_ptr 构造函数看似合理,但容易触发未定义行为或绕过资源管理逻辑。比如 std::shared_ptr 会构造一个空智能指针,但它不持有任何控制块——后续调用 use_count() 返回 0,reset() 行为也和预期不同。
更安全的做法是用默认构造或 std::shared_ptr 显式构造空指针:
std::shared_ptr—— 推荐,语义清晰,控制块不存在,开销最小ptr; std::shared_ptr—— 编译通过,但底层仍调用ptr = nullptr; shared_ptr(nullptr, Deleter{}),多一次空控制块分配(部分实现)- 避免
std::shared_ptr在条件分支里反复构造,可能泄漏控制块内存(某些 libstdc++ 旧版本有此问题)(nullptr)
std::unique_ptr 转移后访问 raw pointer 是常见崩溃源头
一旦 std::unique_ptr 调用 release() 或被移动(std::move(ptr)),它内部的裸指针就置为 nullptr,但原来指向的内存并未立即释放。此时若还保留了该裸指针副本(比如存到某个 int* 变量里),后续解引用就是野指针。
典型错误场景:
- 从
std::unique_ptr拿出p = std::make_unique (); Foo* raw = p.get();,然后执行p.reset();或return std::move(p);—— 此时raw已失效 - 把
p.release()返回的指针交给 C API 后,忘了自己不再负责释放,又在别处 delete 它 - 调试时打印
p.get()地址,误以为“地址还在”就代表对象还活着
std::weak_ptr.lock() 返回空要立刻检查,不能假设一定成功
std::weak_ptr 的存在意义就是应对生命周期不确定的观察场景,lock() 失败不是异常,而是常态。很多同学写成 auto ptr = wp.lock(); ptr->do_something();,一旦 wp 所观察的 shared_ptr 已销毁,ptr 就是空,解引用直接 crash。
正确姿势是每次使用前做空检查:
- 用
if (auto ptr = wp.lock()) { ptr->do_something(); }—— C++17 后推荐,作用域自动限制 - 不要缓存
wp.lock()结果跨函数调用,哪怕只隔一行:auto ptr = wp.lock(); some_func(); ptr->xxx;中间some_func()可能触发销毁 wp.expired()只比lock()少一次引用计数增减,性能差异可忽略,但不能替代lock()后的空判断
自定义 deleter 导致 sizeof(std::shared_ptr) 翻倍很常见
默认 std::shared_ptr 的大小通常是两个指针(8 字节 ×2 = 16 字节,x64)。但只要用了非默认 deleter(比如 lambda、函数对象、std::function),控制块就必须存储该 deleter 实例,导致 shared_ptr 对象本身变大——常见涨到 32 字节甚至更多。
影响不只是内存占用:
- 作为结构体成员时,padding 和 cache line 对齐可能恶化
- 频繁拷贝(如容器中传递)成本上升,尤其在 hot path 上
- lambda 捕获变量会让 deleter 不再是 trivially copyable,某些优化(如 memcpy 替代拷贝构造)失效
- 推荐优先用函数指针形式 deleter:
[](void* p) { free(p); }比[ctx](void* p) { free_with_ctx(p, ctx); }更轻量
复杂点在于:deleter 类型一旦确定,就绑定进模板实例,没法运行时切换。所以一开始选错,后面改起来牵连甚广。
今天关于《千问AI教你C++智能指针使用技巧》的内容介绍就到此结束,如果有什么疑问或者建议,可以在golang学习网公众号下多多回复交流;文中若有不正之处,也希望回复留言以告知!
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
198 收藏
-
415 收藏
-
377 收藏
-
126 收藏
-
265 收藏
-
114 收藏
-
299 收藏
-
225 收藏
-
136 收藏
-
394 收藏
-
319 收藏
-
496 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 485次学习