0

0

C++智能指针循环引用 实际案例与解决方案

P粉602998670

P粉602998670

发布时间:2025-08-29 10:53:01

|

638人浏览过

|

来源于php中文网

原创

使用 weak_ptr 可解决 shared_ptr 循环引用问题。在树形结构中,子节点通过 weak_ptr 指向父节点,避免引用计数无法归零,确保对象正确析构,从而防止内存泄漏。

c++智能指针循环引用 实际案例与解决方案

智能指针是 C++ 中管理动态内存的重要工具std::shared_ptr 通过引用计数自动释放资源,但在某些场景下容易引发循环引用问题,导致内存无法释放。下面通过一个实际案例说明问题,并给出有效解决方案。

实际案例:父子节点间的循环引用

考虑一个树形结构,每个父节点通过 shared_ptr 管理子节点,同时子节点需要访问父节点(例如向上遍历或通知)。常见的做法是子节点也持有一个指向父节点的 shared_ptr

示例代码:

#include 
#include 

struct Node;
using NodePtr = std::shared_ptr;

struct Node {
    int value;
    NodePtr parent;
    NodePtr child;

    Node(int v) : value(v) {
        std::cout << "Node " << value << " created.\n";
    }

    ~Node() {
        std::cout << "Node " << value << " destroyed.\n";
    }
};

构建父子关系:

int main() {
    auto parent = std::make_shared(1);
    auto child  = std::make_shared(2);

    parent->child = child;
    child->parent = parent;  // 循环引用形成

    return 0;
}

运行结果:程序结束时,两个节点的析构函数都未被调用。因为 parent 持有 child 的引用(计数为2),child 又持有 parent 的引用(计数也为2),当作用域结束时,引用计数仅减为1,无法归零,导致内存泄漏。

立即学习C++免费学习笔记(深入)”;

解决方案:使用 weak_ptr 打破循环

关键是将双向引用中的一方改为 std::weak_ptr。它不增加引用计数,只观察对象是否存在,适合用于“从属”关系。

修改子节点中的父节点引用:

Remove.bg
Remove.bg

AI在线抠图软件,图片去除背景

下载

struct Node {
    int value;
    NodePtr child;
    std::weak_ptr parent;  // 改为 weak_ptr

    Node(int v) : value(v) {
        std::cout << "Node " << value << " created.\n";
    }

    ~Node() {
        std::cout << "Node " << value << " destroyed.\n";
    }
};

访问父节点时需临时提升为 shared_ptr

void printParentValue(const NodePtr& node) {
    auto parent = node->parent.lock();  // 提升为 shared_ptr
    if (parent) {
        std::cout << "Parent value: " << parent->value << "\n";
    } else {
        std::cout << "Parent no longer exists.\n";
    }
}

此时再运行程序,main 函数结束时两个节点都能正确析构,内存被释放。

何时使用 weak_ptr

在以下场景中应优先考虑 weak_ptr

  • 观察者模式中,观察者持有被观察者的弱引用
  • 父-子结构中,子节点回指父节点
  • 缓存系统中,避免缓存对象阻止资源释放
  • 回调机制中,防止对象因回调引用无法销毁

总结

循环引用不是 shared_ptr 的缺陷,而是使用方式的问题。关键在于理清对象生命周期的主导关系:生命周期长或主导的一方用 shared_ptr,从属或短暂的一方用 weak_ptr。通过合理使用 weak_ptr,既能保持灵活性,又能避免内存泄漏。

基本上就这些。只要在设计双向关联时多思考引用关系,就能有效规避循环引用问题。

相关专题

更多
Golang 性能分析与pprof调优实战
Golang 性能分析与pprof调优实战

本专题系统讲解 Golang 应用的性能分析与调优方法,重点覆盖 pprof 的使用方式,包括 CPU、内存、阻塞与 goroutine 分析,火焰图解读,常见性能瓶颈定位思路,以及在真实项目中进行针对性优化的实践技巧。通过案例讲解,帮助开发者掌握 用数据驱动的方式持续提升 Go 程序性能与稳定性。

9

2026.01.22

html编辑相关教程合集
html编辑相关教程合集

本专题整合了html编辑相关教程合集,阅读专题下面的文章了解更多详细内容。

53

2026.01.21

三角洲入口地址合集
三角洲入口地址合集

本专题整合了三角洲入口地址合集,阅读专题下面的文章了解更多详细内容。

28

2026.01.21

AO3中文版入口地址大全
AO3中文版入口地址大全

本专题整合了AO3中文版入口地址大全,阅读专题下面的的文章了解更多详细内容。

365

2026.01.21

妖精漫画入口地址合集
妖精漫画入口地址合集

本专题整合了妖精漫画入口地址合集,阅读专题下面的文章了解更多详细内容。

111

2026.01.21

java版本选择建议
java版本选择建议

本专题整合了java版本相关合集,阅读专题下面的文章了解更多详细内容。

3

2026.01.21

Java编译相关教程合集
Java编译相关教程合集

本专题整合了Java编译相关教程,阅读专题下面的文章了解更多详细内容。

15

2026.01.21

C++多线程相关合集
C++多线程相关合集

本专题整合了C++多线程相关教程,阅读专题下面的的文章了解更多详细内容。

9

2026.01.21

无人机驾驶证报考 uom民用无人机综合管理平台官网
无人机驾驶证报考 uom民用无人机综合管理平台官网

无人机驾驶证(CAAC执照)报考需年满16周岁,初中以上学历,身体健康(矫正视力1.0以上,无严重疾病),且无犯罪记录。个人需通过民航局授权的训练机构报名,经理论(法规、原理)、模拟飞行、实操(GPS/姿态模式)及地面站训练后考试合格,通常15-25天拿证。

46

2026.01.21

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号