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

C++ shared_ptr 底层实现分析

智能指针引入的原因:指针管理的困境:内存泄漏, 悬垂指针(指针指向的内存释放或重新分配内存导致),重复释放,野指针(指针未被正确初始化)

shared_ptr 是 C++ 智能指针中最重要的一个,它实现了共享所有权的概念,通过引用计数机制来管理对象的生命周期。下面我将详细分析其底层实现原理。

核心实现机制

shared_ptr 的核心实现通常包含以下几个关键部分:

  1. 控制块(Control Block)

    • 存储引用计数(shared count)

    • 存储弱引用计数(weak count)

    • 存储删除器(deleter)

    • 存储分配器(allocator)

    • 指向被管理对象的指针

  2. 引用计数

    • 每当一个新的 shared_ptr 指向同一对象时,引用计数增加

    • 当 shared_ptr 被销毁或重置时,引用计数减少

    • 当引用计数降为 0 时,删除被管理对象

典型实现结构

template<typename T>
class shared_ptr {
private:T* ptr;                  // 指向被管理对象的指针ControlBlock* control;   // 指向控制块的指针struct ControlBlock {long shared_count;    // 共享引用计数long weak_count;      // 弱引用计数Deleter deleter;      // 删除器// 可能还有其他成员...};public:// 构造函数、析构函数、拷贝控制成员等...
};

关键操作实现

构造函数

template<typename T>
shared_ptr<T>::shared_ptr(T* p) : ptr(p), control(new ControlBlock) {if (control) {control->shared_count = 1;control->weak_count = 0;}
}

拷贝构造函数

template<typename T>
shared_ptr<T>::shared_ptr(const shared_ptr<T>& other) : ptr(other.ptr), control(other.control) {if (control) {++control->shared_count;}
}

析构函数

template<typename T>
shared_ptr<T>::~shared_ptr() {if (control) {--control->shared_count;if (control->shared_count == 0) {// 删除被管理对象control->deleter(ptr);// 如果没有弱引用,删除控制块if (control->weak_count == 0) {delete control;}}}
}

赋值操作符

template<typename T>
shared_ptr<T>& shared_ptr<T>::operator=(const shared_ptr<T>& other) {if (this != &other) {// 减少当前引用计数this->~shared_ptr();// 复制新指针和控制块ptr = other.ptr;control = other.control;if (control) {++control->shared_count;}}return *this;
}

线程安全性

现代 shared_ptr 实现通常保证:

  • 不同 shared_ptr 实例可以被多线程同时访问(即使它们管理同一对象)

  • 同一 shared_ptr 实例的多个成员函数调用需要外部同步

  • 引用计数的增减是原子操作(通常使用原子操作或互斥锁实现)

性能考虑

  1. 内存开销

    • 每个 shared_ptr 对象通常需要存储两个指针(对象指针和控制块指针)

    • 控制块本身也有额外内存开销

  2. 时间开销

    • 引用计数的增减操作需要原子操作,比普通指针操作慢

    • 控制块的动态分配也需要时间

自定义删除器

shared_ptr 支持自定义删除器,这在管理特殊资源时非常有用:

struct FileDeleter {void operator()(FILE* fp) const {if (fp) fclose(fp);}
};shared_ptr<FILE> filePtr(fopen("test.txt", "r"), FileDeleter());

与 weak_ptr 的交互

weak_ptr 也使用相同的控制块,但只增加弱引用计数,不影响共享引用计数。当最后一个 shared_ptr 被销毁时,对象会被删除,但控制块会保留直到所有 weak_ptr 也被销毁。

解决循环引用 

class Node {
public:// shared_ptr<Node> next;  // 这样会导致循环引用weak_ptr<Node> next;      // 使用 weak_ptr 避免循环~Node() { cout << "Node destroyed\n"; }
};void no_circular_reference() {shared_ptr n1 = make_shared<Node>();shared_ptr n2 = make_shared<Node>();n1->next = n2;n2->next = n1;  // 不会造成循环引用
}

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

相关文章:

  • uniapp+vue3+鸿蒙系统的开发
  • WD5018 同步整流降压转换器核心特性与应用,电压12V降5V,2A电流输出
  • MySQL学习——面试版
  • ssl相关命令生成证书
  • LangChain面试内容整理-知识点21:LangSmith 调试与监控平台
  • 职业发展:把工作“玩”成一场“自我升级”的游戏
  • 如何解决pip安装报错ModuleNotFoundError: No module named ‘tkinter’问题
  • webpack相关
  • 基于Matlab的四旋翼无人机动力学PID控制仿真
  • 第五届计算机科学与区块链国际学术会议(CCSB 2025)
  • 大模型训练框架对比
  • CTFMisc之隐写基础学习
  • 重学前端007 --- CSS 排版
  • day22 力扣77.组合 力扣216.组合总和III 力扣17.电话号码的字母组合
  • 异常流程进阶 —— 进出异常时的压栈与出栈
  • LVS集群搭建
  • 【Excel】使用vlookup函数快速找出两列数据的差异项
  • 零基础学Vue3组件化开发
  • 使用Python清理Excel中的空行和单元格内部空行:初学者指南
  • Excel处理控件Aspose.Cells教程:使用 Python 在 Excel 中创建甘特图
  • EP02:【NLP 第二弹】自然语言处理数据
  • Oracle 大页配置use_large_pages 参数解析
  • Antd中使用Table集成 react-resizable实现可伸缩列
  • 高性能上位机界面设计范式:C#与C++/C开发调试无缝衔接
  • AR智能巡检:电力运维的数字化变革
  • Raydium CLMM 协议
  • Kotlin比较接口
  • 安全初级作业2
  • HTTP vs HTTPS
  • RabbitMQ工作模式