c++中weak_ptr如何提升为shared_ptr_c++ lock方法使用【基础】

发布时间 - 2026-01-29 00:00:00    点击率:
weak_ptr.lock()是唯一安全的提升方式,返回shared_ptr:对象存活则增加引用计数并返回有效指针,否则返回空;它是原子操作,线程安全,但必须检查返回值是否为空以避免解引用崩溃。

weak_ptr.lock() 是唯一安全的提升方式

直接用 shared_ptr 构造函数或赋值把 weak_ptr 转成 shared_ptr 会编译失败——weak_ptr 不提供隐式转换,也不接受裸指针构造。唯一合法、线程安全的提升方法就是调用 lock()

它返回一个 shared_ptr:如果原 weak_ptr 指向的对象还活着,就增加引用计数并返回有效指针;如果对象已被释放,就返回空的 shared_ptrptr.get() == nullptr)。

  • lock() 是原子操作,多线程下调用安全
  • 返回的 shared_ptr 和原 shared_ptr 共享同一控制块,引用计数正确同步
  • 不要对 lock() 结果做“先判空再用”的假设性解引用——必须检查返回值

不检查 lock() 返回值会导致崩溃

常见错误是忽略 lock() 可能返回空指针,直接解引用:

std::weak_ptr wp = std::make_shared(42);
// ... 期间原始 shared_ptr 被销毁
auto sp = wp.lock(); // 此时 sp 为空
std::cout << *sp; // 未定义行为:解引用空 shared_ptr → 崩溃或随机值

正确做法始终检查:

  • if (auto sp = wp.lock()) { /* 安全使用 sp */ }
  • 或显式判断:if (sp) { ... }sp.get() != nullptr 等价但更啰嗦
  • 不能只靠 wp.expired() 提前判断——它和 lock() 之间存在竞态窗口(尤其多线程)

lock() 和 expired() 的行为差异与适用场景

expired() 只读取引用计数是否为 0,开销极小;lock() 尝试原子地增加计数,有轻微开销但带回了所有权。

  • 仅需知道“对象是否还存在”(比如日志、统计、跳过处理)→ 用 expired()
  • 需要访问对象内容或延长其生命周期 → 必须用 lock() 并检查结果
  • 二者不是等价替代:即

    使 !wp.expired() 为 true,wp.lock() 仍可能返回空(极端竞态下控制块被销毁的瞬间)

循环引用中 weak_ptr.lock() 的典型误用

在父-子双向持有场景里,子用 weak_ptr 持父是标准解法。但容易在回调中写错:

struct Parent {
    std::shared_ptr child;
};
struct Child {
    std::weak_ptr parent;
    void onEvent() {
        auto p = parent.lock(); // ✅ 正确:每次需要时都 lock
        if (p) p->doSomething();
    }
};
  • 错误:把 lock() 结果缓存为成员变量(如 std::shared_ptr cached_parent)——这又引入了强引用,可能复活已销毁对象或干扰析构顺序
  • 错误:在构造函数里提前 lock() 并保存——此时父对象可能尚未完全构造完毕,引发未定义行为
  • 原则:lock() 应该按需、就近、一次一用

真正容易被忽略的是:lock() 成功不代表对象处于可用状态——比如对象内部已进入析构中途(如虚函数表被清空),此时调用成员函数仍可能崩溃。所以业务逻辑层还需配合状态标记或 RAII 守卫。


# ai  # c++  # 隐式转换  # red  # if  # 成员变量  # 成员函数  # 构造函数  # auto  # 循环  # 指针  # 虚函数  # 线程  # 多线程  # 空指针  # 对象  # 返回值  # 为空  # 的是  # 也不  # 已被  # 是唯一  # 不代表  # 它是  # 要对 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Android滚轮选择时间控件使用详解  ChatGPT常用指令模板大全 新手快速上手的万能Prompt合集  Python函数文档自动校验_规范解析【教程】  mc皮肤壁纸制作器,苹果平板怎么设置自己想要的壁纸我的世界?  Laravel Eloquent访问器与修改器是什么_Laravel Accessors & Mutators数据处理技巧  bing浏览器学术搜索入口_bing学术文献检索地址  Laravel怎么实现模型属性的自动加密  Win11怎么更改系统语言为中文_Windows11安装语言包并设为显示语言  php结合redis实现高并发下的抢购、秒杀功能的实例  Laravel如何创建自定义Facades?(详细步骤)  Laravel与Inertia.js怎么结合_使用Laravel和Inertia构建现代单页应用  Win11关机界面怎么改_Win11自定义关机画面设置【工具】  微信小程序 wx.uploadFile无法上传解决办法  canvas 画布在主流浏览器中的尺寸限制详细介绍  安克发布新款氮化镓充电宝:体积缩小 30%,支持 200W 输出  Laravel怎么自定义错误页面_Laravel修改404和500页面模板  Laravel如何使用Gate和Policy进行权限控制_Laravel权限判定与策略规则配置  历史网站制作软件,华为如何找回被删除的网站?  lovemo网页版地址 lovemo官网手机登录  laravel怎么实现图片的压缩和裁剪_laravel图片压缩与裁剪方法  桂林网站制作公司有哪些,桂林马拉松怎么报名?  Swift开发中switch语句值绑定模式  VIVO手机上del键无效OnKeyListener不响应的原因及解决方法  php做exe能调用系统命令吗_执行cmd指令实现方式【详解】  Laravel如何实现本地化和多语言支持?(i18n教程)  高配服务器限时抢购:企业级配置与回收服务一站式优惠方案  如何在IIS中配置站点IP、端口及主机头?  Win11怎么查看显卡温度 Win11任务管理器查看GPU温度【技巧】  javascript中的数组方法有哪些_如何利用数组方法简化数据处理  如何用IIS7快速搭建并优化网站站点?  Laravel Session怎么存储_Laravel Session驱动配置详解  google浏览器怎么清理缓存_谷歌浏览器清除缓存加速详细步骤  Laravel如何实现图片防盗链功能_Laravel中间件验证Referer来源请求【方案】  Chrome浏览器标签页分组怎么用_谷歌浏览器整理标签页技巧【效率】  Python并发异常传播_错误处理解析【教程】  Laravel如何实现全文搜索功能?(Scout和Algolia示例)  个人网站制作流程图片大全,个人网站如何注销?  高端云建站费用究竟需要多少预算?  Laravel如何集成第三方登录_Laravel Socialite实现微信QQ微博登录  SQL查询语句优化的实用方法总结  HTML透明颜色代码怎么让下拉菜单透明_下拉菜单透明背景指南【技巧】  浅析上传头像示例及其注意事项  详解免费开源的.NET多类型文件解压缩组件SharpZipLib(.NET组件介绍之七)  如何制作一个表白网站视频,关于勇敢表白的小标题?  详解Nginx + Tomcat 反向代理 如何在高效的在一台服务器部署多个站点  如何在 Telegram Web View(iOS)中防止键盘遮挡底部输入框  制作网站软件推荐手机版,如何制作属于自己的手机网站app应用?  深圳网站制作公司好吗,在深圳找工作哪个网站最好啊?  Laravel如何升级到最新版本?(升级指南和步骤)  微信小程序制作网站有哪些,微信小程序需要做网站吗?