shared_ptr与weak_ptr配合可避免循环引用并确保安全访问。shared_ptr共享所有权,延长对象生命周期,而weak_ptr不增加引用计数,仅观察对象是否存在。当存在相互引用场景(如父子节点)时,使用weak_ptr打破循环。访问前必须调用lock()获取shared_ptr,若返回非空则可在if块内安全操作,从而防止访问已释放对象。该机制结合自动内存管理与线程安全的lock操作,有效避免内存泄漏和悬空指针问题。

在C++中,shared_ptr 和 weak_ptr 配合使用可以有效避免循环引用并实现安全的对象访问。当多个对象通过 shared_ptr 相互持有强引用时,可能导致内存无法释放。weak_ptr 提供了一种“观察”资源的方式,不会增加引用计数,从而打破循环。
shared_ptr 与 weak_ptr 的基本关系
shared_ptr 表示对资源的共享所有权,只要有一个 shared_ptr 存在,对象就不会被销毁。而 weak_ptr 是对 shared_ptr 的弱引用,它不控制对象的生命周期,仅能临时“观测”对象是否存在。
使用 weak_ptr 访问对象前,必须先将其转换为 shared_ptr,通常通过 lock() 方法实现:
- 调用 weak_ptr::lock() 返回一个 shared_ptr
- 如果原对象已被释放,返回的 shared_ptr 为空
- 只有在非空情况下才可安全访问对象
防止循环引用的典型场景
父子节点或观察者模式中容易出现 shared_ptr 循环引用。例如,父节点持有子节点的 shared_ptr,子节点若也持有父节点的 shared_ptr,就会导致两者都无法释放。
立即学习“C++免费学习笔记(深入)”;
解决方案是让子节点使用 weak_ptr 持有父节点:
struct Parent;
struct Child {
std::weak_ptr parent;
void do_something() {
auto p = parent.lock();
if (p) {
// 安全访问父节点
} else {
// 父节点已释放
}
}
};
线程安全与正确使用模式
虽然 weak_ptr::lock() 是线程安全的,但检查和使用之间存在时间窗口,直接判断 weak_ptr 是否为空不可靠。正确做法始终是调用 lock() 获取 shared_ptr 再操作。
常见安全访问模式:
- 任何时候访问 weak_ptr 所指对象,都应先 lock() 得到 shared_ptr
- 在 if 条件中检查返回的 shared_ptr 是否有效
- 在 if 块内使用该 shared_ptr,确保对象生命周期被延长
基本上就这些。合理组合 shared_ptr 和 weak_ptr,既能保证资源自动管理,又能避免内存泄漏和访问已释放对象的问题。关键在于理解 weak_ptr 只是“观察者”,真正访问前必须升级为 shared_ptr。










