C++11如何使用std::weak_ptr解决循环引用问题(如何使用.循环.引用.解决.weak_ptr...)

wufei123 发布于 2025-09-17 阅读(8)
循环引用指两个对象互相持有对方的shared_ptr,导致引用计数无法归零而内存泄漏;使用weak_ptr可打破循环,因其不增加引用计数,仅观察对象是否存在,从而确保正确析构。

c++11如何使用std::weak_ptr解决循环引用问题

在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 Post AI

博客文章AI生成器

Post AI50 查看详情 Post AI 访问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++初学者如何实现简单投票系统

标签:  如何使用 循环 引用 

发表评论:

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