
在C++11中,std::shared_ptr通过引用计数自动管理对象生命周期,但当两个对象互相持有对方的std::shared_ptr时,会形成循环引用,导致内存无法释放。这时就需要用std::weak_ptr打破循环。
什么是循环引用假设有两个类A和B,A持有一个指向B的std::shared_ptr,同时B也持有一个指向A的std::shared_ptr:
class B; // 前向声明class A {
public:
std::shared_ptr<B> ptr;
~A() { std::cout << "A destroyed\n"; }
};
class B {
public:
std::shared_ptr<A> ptr;
~B() { std::cout << "B destroyed\n"; }
};
// 使用:
auto a = std::make_shared<A>();
auto b = std::make_shared<B>();
a->ptr = b;
b->ptr = a; // 循环引用形成
此时,a和b的引用计数都为2。离开作用域后,各自的引用计数减1,但仍为1,析构函数不会被调用,造成内存泄漏。
使用std::weak_ptr打破循环解决方法是让其中一个类使用std::weak_ptr,它不增加引用计数,只“观察”对象是否存在。
class B; // 前向声明class A {
public:
std::shared_ptr<B> ptr;
~A() { std::cout << "A destroyed\n"; }
};
class B {
public:
std::weak_ptr<A> ptr; // 改为 weak_ptr
~B() { std::cout << "B destroyed\n"; }
};
// 使用:
auto a = std::make_shared<A>();
auto b = std::make_shared<B>();
a->ptr = b;
b->ptr = a; // 不增加 a 的引用计数
这样,a的引用计数为1(只有外部变量持有),b的引用计数也为1。当离开作用域,a和b都会被正确析构。
Post AI
博客文章AI生成器
50
查看详情
访问weak_ptr指向的对象
由于std::weak_ptr不保证对象仍然存在,访问前必须检查:
if (auto locked = b->ptr.lock()) {// locked 是 std::shared_ptr<A>
std::cout << "A is still alive\n";
} else {
std::cout << "A has been destroyed\n";
}
也可以用
expired()判断是否过期,但
lock()更安全,因为它同时加锁并返回共享指针。
基本上就这些。关键是在可能形成闭环的地方用std::weak_ptr替代std::shared_ptr,保持逻辑关系又不干扰内存回收。常见于父子节点、观察者模式、缓存等场景。
以上就是C++11如何使用std::weak_ptr解决循环引用问题的详细内容,更多请关注知识资源分享宝库其它相关文章!
相关标签: c++ 解决方法 作用域 red if 析构函数 auto 循环 指针 class public 对象 作用域 大家都在看: C++循环优化与算法选择技巧 C++联合体类型转换 安全类型转换方法 C++对象生命周期与内存分配关系 C++11如何使用范围for循环遍历容器 C++初学者如何实现简单投票系统






发表评论:
◎欢迎参与讨论,请在这里发表您的看法、交流您的观点。