当前位置: 首页 > news >正文

c++弱指针实现原理

在 C++ 中,弱指针(std::weak_ptr)是一种特殊的智能指针,其核心目标是‌解决 std::shared_ptr 的循环引用问题‌,同时不增加对象的引用计数。它的实现原理基于与 std::shared_ptr 共享的 ‌控制块(Control Block)‌,并通过 ‌弱引用计数(Weak Reference Count)‌ 管理资源生命周期。

1. 弱指针的设计目标

  • 打破循环引用‌:当两个 std::shared_ptr 互相引用时,引用计数无法归零,导致内存泄漏。std::weak_ptr 不增加引用计数,允许安全观测对象是否存在。
  • 临时访问资源‌:通过 lock() 方法临时获取 std::shared_ptr,确保访问时对象存活。

2. 核心实现原理

(1) 控制块(Control Block)
  • 数据结构‌:
    每个由 std::shared_ptr 管理的对象会关联一个 ‌控制块‌,包含以下信息:

    • 强引用计数(Strong Ref Count)‌:当前 std::shared_ptr 的引用数量。
    • 弱引用计数(Weak Ref Count)‌:当前 std::weak_ptr 的引用数量。
    • 对象指针‌(若强引用计数 > 0,否则为 nullptr)。
    • 删除器(Deleter)‌ 和 ‌分配器(Allocator)‌(可选)。
  • 生命周期‌:

    • 当 ‌强引用计数归零‌ 时,对象被销毁(调用析构函数并释放内存)。
    • 当 ‌弱引用计数归零‌ 时,控制块自身被释放。
(2) std::weak_ptr 的构造
  • 从 std::shared_ptr 构造‌:

std::shared_ptr<int> sp = std::make_shared<int>(42);
std::weak_ptr<int> wp(sp);  // 不增加强引用计数,但增加弱引用计数
    • wp 共享 sp 的控制块,弱引用计数 +1。
  • 从另一个 std::weak_ptr 构造‌:

std::weak_ptr<int> wp2(wp);  // 弱引用计数 +1
    (3) std::weak_ptr 的使用
    • lock() 方法‌:
      尝试将 std::weak_ptr 提升为 std::shared_ptr

    if (auto sp = wp.lock()) {  // 若对象存活,强引用计数 +1
        // 安全使用 sp
    }
    
      • 若对象已销毁(强引用计数为 0),返回空的 std::shared_ptr
    • expired() 方法‌:
      快速检查对象是否存活(无需创建 std::shared_ptr):

    if (!wp.expired()) {  // 检查强引用计数是否 > 0
        // 对象存活
    }
    

    3. 内部实现细节

    (1) 控制块的内存管理
    • std::shared_ptr 构造时‌:
      若首次创建 std::shared_ptr,动态分配控制块,强引用计数初始化为 1,弱引用计数初始化为 1(因为 std::shared_ptr 自身也持有一个弱引用)。

    • std::weak_ptr 构造时‌:
      弱引用计数 +1,但不影响强引用计数。

    • std::shared_ptr 析构时‌:
      强引用计数 -1。若强引用计数归零:

      1. 销毁对象(调用析构函数并释放内存)。
      2. 若弱引用计数也为 0‌,释放控制块;否则保留控制块供 std::weak_ptr 查询。
    • std::weak_ptr 析构时‌:
      弱引用计数 -1。若弱引用计数归零且强引用计数已为 0,释放控制块。

    (2) 线程安全性
    • 引用计数的原子操作‌:
      控制块的引用计数(强/弱)通过原子操作(如 std::atomic)实现线程安全。

    4. 示例:解决循环引用

    #include <memory>
    class Node {
    public:
        std::shared_ptr<Node> next;
        std::weak_ptr<Node> prev;  // 使用 weak_ptr 打破循环引用
    };
    
    int main() {
        auto node1 = std::make_shared<Node>();
        auto node2 = std::make_shared<Node>();
        node1->next = node2;      // node2 强引用计数 = 2
        node2->prev = node1;      // node1 弱引用计数 = 1
        // node1 和 node2 的强引用计数最终可归零,正确释放内存
    }
    

    5. 性能与限制

    • 性能开销‌:

      • std::weak_ptr 的创建、销毁和 lock() 涉及原子操作,略慢于裸指针。
      • 控制块占用额外内存(通常为 2~3 个指针大小)。
    • 使用限制‌:

      • 必须通过 lock() 获取 std::shared_ptr 后才能访问对象。
      • 无法直接访问对象的原始指针(需先调用 lock())。

    总结

    std::weak_ptr 通过 ‌共享控制块‌ 和 ‌分离强/弱引用计数‌ 的机制,实现了对 std::shared_ptr 管理对象的安全观测。其核心价值在于:

    1. 打破循环引用‌,避免内存泄漏。
    2. 临时访问资源‌,确保访问时对象存活。
      它是现代 C++ 内存管理中不可或缺的工具,尤其适用于观察者模式、缓存管理等场景。

    http://www.dtcms.com/a/99177.html

    相关文章:

  • Python小练习系列 Vol.5:数独求解(经典回溯 + 剪枝)
  • Linux之基础知识
  • 深度学习处理时间序列(5)
  • 《新凯来:半导体设备制造领域的“国家队”》
  • 【愚公系列】《高效使用DeepSeek》039-政务工作辅助
  • LeetCode 2360.图中的最长环:一步一打卡(不撞南墙不回头) - 通过故事讲道理
  • Redis延时队列在订单超时未报到场景的应用分享
  • 【数据结构】二叉树 — 经典OJ面试题剖析!!!
  • 关于 websocket协议的理解
  • 001 - 前缀和算法:从原理到实战,一文讲透区间和问题
  • 谈谈Minor GC、Major GC和Full GC
  • Java——数组
  • RSA 简介及 C# 和 js 实现【加密知多少系列_4】
  • .NET开发基础知识11-20
  • Lavazza拉瓦萨亮相上海樱花节,增色海派咖啡风情
  • rbpf虚拟机-汇编和反汇编器
  • OpenCV、YOLO与大模型的区别与关系
  • Web开发:数据的加密和解密
  • Python之函数
  • 图片RGBA像素值提取工具v1.0.0发布
  • 【原理系列】计算机组成原理-第一遍阅读总结
  • Flutter_学习记录_AppBar中取消leading的占位展示
  • Python:日志模块操作及基本配置,日志格式化输出
  • Mybatis源码 插件机制
  • Vue3 项目通过 docxtemplater 插件动态渲染 .docx 文档(带图片)预览,并导出
  • 人工智能与软件工程结合的发展趋势
  • 一些常用开发软件下载地址
  • [Python]如何利用Flask搭建一個Web服務器,並透過Ngrok訪問來實現LINE Bot功能?
  • MySQL数据库的操作(mybatis)
  • Spring学习笔记06——bean、java bean、spring bean、POJO几个概念讲解