循环引用指两个对象的shared_ptr相互持有,导致引用计数无法归零而内存泄漏;解决方法是用weak_ptr打破闭环,如父节点用shared_ptr,子节点用weak_ptr指向父节点,避免析构失败。

使用 shared_ptr 时,循环引用是一个常见问题,尤其在涉及双向关系的场景中,比如父子节点、观察者模式等。当两个对象通过 shared_ptr 相互持有对方时,引用计数无法归零,导致内存泄漏。
什么是循环引用?
假设有两个类 A 和 B,A 中有一个 shared_ptr,B 中也有一个 shared_ptr。如果它们相互赋值并形成闭环:
struct A { std::shared_ptr ptr; ~A() { std::cout << "A destroyedn"; }};struct B { std::shared_ptr ptr; ~B() { std::cout << "B destroyedn"; }};auto a = std::make_shared();auto b = std::make_shared();a->ptr = b;b->ptr = a;
此时,a 和 b 的引用计数都为 2。离开作用域后,各自释放一次,引用计数变为 1,但不会调用析构函数,造成内存泄漏。
使用 weak_ptr 打破循环
weak_ptr 是专门设计用来解决这个问题的智能指针。它不增加引用计数,只是“观察” shared_ptr 管理的对象。
立即学习“C++免费学习笔记(深入)”;
修改上面的例子,把其中一个方向改为 weak_ptr:
struct B { std::weak_ptr ptr; // 改为 weak_ptr ~B() { std::cout << "B destroyedn"; }};
这样,B 持有的是 A 的弱引用,不会增加其引用计数。当 a 离开作用域,A 的引用计数减到 0,被正确销毁;随后 b 销毁,B 也被释放。
访问 weak_ptr 时需先检查对象是否还存在:
if (auto locked = b.ptr.lock()) { // 使用 locked(返回 shared_ptr)} else { // 对象已被释放}
实际使用建议
在设计类关系时,明确“所有权”:
用 shared_ptr 表示共享所有权或强引用 用 weak_ptr 表示非拥有型引用,如父-子结构中的反向指针、缓存、观察者列表等 避免两个 shared_ptr 在没有外部干预的情况下互相持有
例如,在树结构中,父节点用 shared_ptr 指向子节点,子节点用 weak_ptr 指向父节点,这样能安全释放整棵树。
基本上就这些。只要在可能形成闭环的地方主动用 weak_ptr 断开一环,就能有效防止循环引用。不复杂但容易忽略。
以上就是C++ shared_ptr怎么解决循环引用_C++智能指针使用与循环引用解决方案的详细内容,更多请关注创想鸟其它相关文章!
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1484342.html
微信扫一扫
支付宝扫一扫