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 强引用计数 = 2node2->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.mzph.cn/bicheng/75006.shtml

    如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

    相关文章

    【ManiSkill】环境success条件和reward函数学习笔记

    1. “PickCube-v1” info["success"]&#xff1a;用于指示任务是否成功完成 布尔型张量&#xff0c;在环境的evaluate()方法中计算并返回&#xff1a; "success": is_obj_placed & is_robot_static这确保了机器人不仅能将物体准确放置在目标位置&am…

    用空闲时间做了一个小程序-二维码生成器

    一直在摸鱼中赚钱的大家好呀~ 先向各位鱼友们汇报一下情况&#xff0c;目前小程序已经有900的鱼友注册使用过。虽然每天都有新的鱼友注册&#xff0c;但是鱼友增长的还很缓慢。自从国庆前的文字转语音的工具上线到现在已经将近有1个月没有更新小程序了。但是今天终终终终终于又…

    31天Python入门——第14天:异常处理

    你好&#xff0c;我是安然无虞。 文章目录 异常处理1. Python异常2. 异常捕获try-except语句捕获所有的异常信息获取异常对象finally块 3. raise语句4. 自定义异常5. 函数调用里面产生的异常补充练习 异常处理 1. Python异常 Python异常指的是在程序执行过程中发生的错误或异…

    PyQt6实例_批量下载pdf工具_使用pyinstaller与installForge打包成exe文件

    目录 前置&#xff1a; 步骤&#xff1a; step one 准备好已开发完毕的项目代码 step two 安装pyinstaller step three 执行pyinstaller pdfdownload.py&#xff0c;获取初始.spec文件 step four 修改.spec文件&#xff0c;将data文件夹加入到打包程序中 step five 增加…

    Axure项目实战:智慧城市APP(完整交互汇总版)

    亲爱的小伙伴&#xff0c;在您浏览之前&#xff0c;烦请关注一下&#xff0c;在此深表感谢&#xff01; 课程主题&#xff1a;智慧城市APP 主要内容&#xff1a;主功能&#xff08;社保查询、医疗信息、公交查询等&#xff09;、活动、消息、我的页面汇总 应用场景&#xff…

    Appium Inspector使用教程

    1.下载最新版本 https://github.com/appium/appium-inspector/releases 2.本地启动一个Appium服务 若Android SDK已安装Appium服务&#xff0c;则在任意terminal使用appium启动服务即可 3.Appium Inspector客户端配置连接到Appium服务 Configuring and Starting a Session…

    Pycharm(七):几个简单案例

    一.剪刀石头布 需求&#xff1a;和电脑玩剪刀石头布游戏 考察点&#xff1a;1.随机数&#xff1b;2.判断语句 import random # numrandom.randint(1,3) # print(num) # print(**30) #1.录入玩家手势 playerint(input(请输入手势&#xff1a;&#xff08;1.剪刀 2.石头 3&…

    Python Cookbook-4.13 获取字典的一个子集

    任务 你有一个巨大的字典&#xff0c;字典中的一些键属于一个特定的集合&#xff0c;而你想创建一个包含这个键集合及其对应值的新字典。 解决方案 如果你不想改动原字典: def sub_dict(somedict,somekeys,default None):return dict([(k, somedict.get(k,default)) for k…

    VMware Ubuntu 网络配置全攻略:从断网到畅通无阻

    一、网络连接模式选择&#xff08;先搞懂原理&#xff09; VMware提供三种网络模式&#xff0c;就像手机的不同网络套餐&#xff1a; 模式适用场景特点类比NAT个人上网/新手首选虚拟机共享主机IP&#xff0c;能上网但隐身家用WiFi桥接服务器/需要被局域网访问虚拟机会获得独立…

    链表(C++)

    这是本人第二次学习链表&#xff0c;第一次学习链表是在大一上的C语言课上&#xff0c;首次接触&#xff0c;感到有些难&#xff1b;第二次是在大一下学习数据结构时&#xff08;就是这次&#xff09;&#xff0c;使用C再次理解链表。同时&#xff0c;这也是开启数据结构学习写…

    【SPP】蓝牙串口协议应用层深度解析:从连接建立到实战开发

    目录 一、SPP应用层协议框架与角色模型 1.1 分层协议栈模型 1.2 设备角色模型&#xff08;DevA 与 DevB 交互&#xff09; 二、连接建立流程&#xff1a;从 SDP 到 RFCOMM 2.1 服务发现&#xff08;SDP&#xff09;流程&#xff08;SDP 记录关键参数&#xff09; 2.2 连接…

    Giteki 认证:无线产品进入日本市场的关键保障

    目录 适用产品认证范围 认证项目及技术要求 认证流程 认证周期 与其他认证的对比 常见问题 注意事项 Giteki 认证&#xff0c;其名称来源于日本语 “技適マーク”&#xff0c;罗马字拼写为 “GITEKI” &#xff0c;在行业内也常被称为 Telec 认证、MIC 认证、RF 认证或技…

    Ubuntu24.04 配置远程桌面服务

    一&#xff1a;安装 sudo apt update sudo apt install vino 二&#xff1a;设置 gsettings set org.gnome.Vino require-encryption false # 关闭加密&#xff08;某些 VNC 客户端不支持加密&#xff09; gsettings set org.gnome.Vino prompt-enabled false # 关闭连接…

    人工智能与软件工程结合的发展趋势

    AI与软件工程的结合正在深刻改变软件开发的流程、工具和方法&#xff0c;其发展方向涵盖了从代码生成到系统维护的整个生命周期。以下是主要的发展方向和技术趋势&#xff1a; 1. 软件架构体系的重构 从“面向过程”到“面向目标”的架构转型&#xff1a; AI驱动软件设计以目标…

    转发和重定向的区别详解

    转发&#xff08;Forward&#xff09;和重定向&#xff08;Redirect&#xff09;是 Web 开发中两种常用的请求处理方式&#xff0c;主要用于将客户端请求从一个资源转移到另一个资源。它们在实现机制、行为表现和应用场景上有显著区别&#xff0c;以下是对两者的详细解析&#…

    python专题1-----判断一个变量是否是字符串类型

    在 Python 中&#xff0c;可以使用 isinstance() 函数来判断一个变量是否是字符串类型。字符串在 Python 中是以 str 类型表示的。下面是一些示例代码&#xff0c;展示如何判断一个变量是否是字符串类型&#xff1a; # 示例变量 var1 "Hello, World!" var2 12345 …

    软件工程之需求工程(需求获取、分析、验证)

    一、需求获取&#xff08;Requirements Elicitation&#xff09; 1. 定义与目标 需求获取是通过与用户、利益相关者等交互&#xff0c;识别并捕获系统需求的过程&#xff0c;目标是明确用户意图与业务目标&#xff0c;避免后期因需求偏差导致返工。 2. 主要方法 问卷法&…

    Java简单生成pdf

    生成这样的PDF 直接上代码 public static void main(String[] args) {String logoPath "Q:\\IdeaWork\\Demo\\src\\main\\webapp\\images\\logo.jpg"; // 替换为实际路径String baseDir "E:/Demo/TEST/problem/Generate"; // 基础目录int year 2025; //…

    k8s存储介绍(六)StorangeClass

    一、Kubernetes 存储类&#xff08;StorageClass&#xff09;详解 1. 什么是 StorageClass&#xff1f; 在 Kubernetes 中&#xff0c;StorageClass&#xff08;存储类&#xff09;是一种用于动态创建 PersistentVolume&#xff08;PV&#xff09;的资源对象。它允许管理员根…

    C++:allocator类(动态数组续)

    1.为什么需要 allocator&#xff1f; 在 C 中&#xff0c;动态内存管理通常通过 new 和 delete 完成&#xff1a; int* p new int; // 分配内存 构造对象 delete p; // 析构对象 释放内存 但 new 和 delete 有两个问题&#xff1a; 耦合性&#xff1a;将内…