std::weak_ptr本身不解决循环引用,它只是打破强引用链的工具;真正起作用的是用std::weak_ptr替换其中一端的std::shared_ptr,使引用计数能自然归零。

直接说结论:std::weak_ptr 本身不“解决”循环引用,它只是打破强引用链的工具;真正起作用的是用 std::weak_ptr 替换掉其中一端的 std::shared_ptr,让引用计数能自然归零。
为什么循环引用会导致内存泄漏
当两个对象互相用 std::shared_ptr 持有对方时,它们的引用计数永远 ≥1,即使外部所有 shared_ptr 都已销毁,析构函数也不会被调用——因为彼此还在“强持有”对方。
典型场景:树节点父子关系、观察者模式中的回调绑定、双向链表节点。
用 std::weak_ptr 打破哪一端?
必须选“非拥有方”那一端。比如:
立即学习“C++免费学习笔记(深入)”;
- 父节点拥有子节点 → 子节点用
std::weak_ptr指向父节点 - 观察者注册到被观察者 → 被观察者用
std::weak_ptr存储观察者(避免观察者销毁后仍被强引用) - 链表中,前驱/后继任选其一用
weak_ptr(通常后继用shared_ptr,前驱用weak_ptr)
关键判断依据:哪个对象的生命周期**逻辑上不依赖**于另一个。不能随便换,否则 lock() 可能返回空 shared_ptr,引发未定义行为。
weak_ptr::lock() 不是“自动转 shared_ptr”,而是“安全尝试”
std::weak_ptr 不能直接解引用,必须先调用 lock() 获取一个临时 std::shared_ptr。这个操作是线程安全的,但返回值可能为空——说明目标对象已被销毁。
auto parent_ptr = child.parent.lock();
if (parent_ptr) {
// 安全使用 parent_ptr
parent_ptr->do_something();
} else {
// 父节点已析构,跳过或处理异常路径
}
常见错误:
- 直接写
*child.parent→ 编译失败 - 写
child.parent.lock()->do_something()→ 空指针解引用崩溃 - 把
lock()结果存在成员变量里长期持有 → 又变相制造了强引用,白用了weak_ptr
别忽略自赋值和多线程竞争
在多线程环境中,lock() 成功只保证“那一刻对象还活着”,不代表后续访问安全。如果对象在 lock() 后立即析构(且无其他 shared_ptr 存活),再次访问仍是 UB。
更稳健的做法是:一次 lock(),把结果存为局部 shared_ptr,所有操作都在该副本生命周期内完成。
另外,注意 weak_ptr 自身可被拷贝、赋值,但赋值不改变所指向对象的引用计数;而 shared_ptr 赋值会增加计数——这点常被误用于“绕过 weak_ptr 设计初衷”。
循环引用不是靠加个 weak_ptr 就自动消失的,得想清楚谁拥有谁、谁先销毁、谁需要感知对方是否存活。漏掉任何一个环节,weak_ptr 就只是个摆设。











