[C++] 智能指针 进阶

标题:[C++] 智能指针 进阶
水墨不写bug

在这里插入图片描述


在很久之前我们探讨了智能指针的浅显认识,接下来会更加深入,从源码角度认识智能指针,从而了解智能指针的设计原理,并应用到以后的工作项目中。

本文将会按照C++智能指针的发展历史,回顾智能指针的各个版本的设计。后半部分将会模拟实现最常考的shared_ptr的功能。


一、C++智能指针的发展历程

1. C++98时代:auto_ptr的尝试与缺陷

背景:C++早期依赖手动new/delete,稍微有管理不当比如忘记释放,或者有异常抛出退出调用的函数栈,但是堆区的指针却找不到了),就可能导致内存泄漏。C++98引入auto_ptr,首次尝试自动化资源管理。
机制:基于RAII(资源获取即初始化),在构造时获取内存,在析构时自动释放内存
问题

  • 隐式所有权转移:复制auto_ptr时,原指针变为nullptr,导致难以追踪的悬空指针。
  • 不兼容容器:因拷贝语义异常,无法安全用于STL容器。

示例

auto_ptr<int> p1(new int(32));
auto_ptr<int> p2 = p1; // p1变为nullptr,后续使用p1导致未定义行为

结局:最终C++11弃用,C++17移除。因此为了代码的逻辑性和健壮性,以及可移植性,非常不建议使用auto_ptr


2. Boost库的智能指针

背景:C++社区通过Boost库探索更健壮的解决方案,后来这些方案被C++委员会收录到了C++11
关键类型

  • scoped_ptr:禁止拷贝,严格独占所有权,轻量且安全。
  • shared_ptr:引用计数实现共享所有权,解决多所有者场景。
  • weak_ptr:打破shared_ptr循环引用,防止内存泄漏。

影响:直接为C++11智能指针奠定基础。


3. C++11:C++划时代的进步

核心类型

  • unique_ptr:取代auto_ptr,独占所有权,支持移动语义,禁止拷贝,可管理数组(unique_ptr<T[]>)。
    unique_ptr<int> p1(new int(10));
    unique_ptr<int> p2 = std::move(p1); // 显式所有权转移
    
  • shared_ptr:引用计数共享资源,线程安全但性能有开销。
    shared_ptr<A> a = make_shared<A>();
    shared_ptr<A> b = a; // 引用计数增至2
    
  • weak_ptr:可通过其获取对应的shared_ptr资源,不增加计数,需通过lock()获取临时shared_ptr
    weak_ptr<A> w = a;
    if (shared_ptr<A> tmp = w.lock()) { /* 使用tmp */ }
    

改进

  • 弃用auto_ptr,推荐unique_ptr
  • 引入make_shared:合并控制块与对象内存分配,提升性能与异常安全。

4. C++14:完善与便利性增强

  • make_unique:填补make_shared的对称性,安全构造unique_ptr
    auto p = make_unique<int>(20); // 避免显式new
    
  • shared_ptr增强:支持自定义删除器与分配器,更灵活的资源管理。

后期的C++17用法不多常见,在此就不再赘述了。

二、智能指针的模拟实现

1、unique_ptr


template<class T>
class UniquePtr
{UniquePtr<T>& operator=(const UniquePtr<T>&) = delete;//删除拷贝构造和赋值重载UniquePtr(const UniquePtr<T>&) = delete;//这是unique_ptr的标志性的特点
public:UniquePtr(T* ptr):_ptr(ptr)//浅拷贝//这就要求我们需要在外部new出堆区的空间,然后传递给智能指针来管理{}~UniquePtr(){if(_ptr)//析构delete _ptr;}T& operator*()//重载*和->让智能指针(类)像普通指针(自定义类型)一样使用{return *_ptr;}T* operator->(){return ptr;}
private:T* _ptr;
};

2、shared_ptr


template<class T>
class SharedPtr
{
public:SharedPtr(T* ptr):_ptr(ptr)//要求从外部传一个指向堆区的指针,_refcount(new int{1})//开辟在堆上,便于所有的管理同一资源的智能指针对象维护,_pmtx(new std::mutex)//同样的原因{}SharedPtr(const SharedPtr<T>& obj):_ptr(obj._ptr)//只需要浅拷贝,然后引用计数++,_refcount(obj._refcount),_pmtx(obj._pmtx){AddRef();}SharedPtr<T>& operator=(const SharedPtr<T>& obj){if(obj._ptr != _ptr)//不能给自身赋值{Release();//释放掉被赋值的智能指针对象的原有资源//浅拷贝_ptr = obj._ptr;//两个指针指向同一个T*对象_refcount = obj._refcount;//两个指针指向同一个int*_pmtx = obj._pmtx;//两个指针指向同一个mutex*的锁AddRef();}}void AddRef(){//增加引用计数需要加锁_pmtx->lock();(*_refcount)++;_pmtx->unlock();}void Release(){_pmtx->lock();bool f = false;if(--(*_refcount) == 0 && _ptr){f = true;delete _ptr;delete _refcount;}_pmtx->unlock();//判断是否资源已经被释放,资源已经被释放,则释放锁if(f == true){delete _pmtx;}}~SharedPtr(){Release();}int UseCount(){return *_refcount;}T& operator*()//重载运算符{return *_ptr;}T* operator->(){return _ptr;}T* GetPtr(){return _ptr;}
private:T* _ptr;//指向堆区资源的裸指针int* _refcount;//开辟在堆区的引用计数std::mutex* _pmtx;//堆区的锁,考虑线程安全//如果设置在栈上,那么每一个对象都有一个独立的引用计数//每一次自增自减都需要对同一引用计数的所有对象维护,非常难以维护
};

3、weak_ptr

库中的wear_ptr的实现需要结合起来shared_ptr,shared_ptr内部含有两个引用计数,一个是记录管理资源的shared_ptr(占用引用计数)的个数,一个记录指向资源的weak_ptr(不占用引用计数)的个数。
在这里,为了简便而言,我们实现一个简化版本的weak_ptr,对上面的知识理解即可。


// 简化版本的weak_ptr实现
template<class T>
class WeakPtr
{
public:WeakPtr():_ptr(nullptr){}WeakPtr(const SharedPtr<T>& sp):_ptr(sp.get()){}WeakPtr<T>& operator=(const SharedPtr<T>& sp){_ptr = sp.get();return *this;}T& operator*(){return *_ptr;}T* operator->(){return _ptr;}
private:T* _ptr;
};

~完
转载请注明出处

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/43142.html

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

相关文章

Linux系统中应用端控制串口的基本方法

在编写应用程序的时候,我们需要将上面提到的串口参数如波特率、数据位、奇偶校验位、停止位 等设置好,通讯双方才能在预定好的参数下相互通讯。哪里设置参数,通过什么设置参数,还有 如何设置这些参数,接下来我们分别来实现。 1:linux系统中termios结构体用法 POS…

高效加盖骑缝章:PDF文件处理的实用解决方案

今天为大家推荐一款实用的PDF骑缝章加盖工具&#xff0c;它能一键生成精准骑缝章&#xff0c;无需复杂操作&#xff0c;极大提升办公效率。 01 软件介绍 这款软件就是PDF加盖骑缝章&#xff1a; 绿色免费且无需安装&#xff0c;解压即用&#xff0c;不占用系统资源 支持Windo…

LLM架构解析:NLP基础(第一部分)—— 模型、核心技术与发展历程全解析

本专栏深入探究从循环神经网络&#xff08;RNN&#xff09;到Transformer等自然语言处理&#xff08;NLP&#xff09;模型的架构&#xff0c;以及基于这些模型构建的应用程序。 本系列文章内容&#xff1a; NLP自然语言处理基础&#xff08;本文&#xff09;词嵌入&#xff0…

黑马点评Feed流推送帖子zset实现

什么是Feed流? 顾名思义就是投喂流 传统的信息查找方式用户需要手动去搜寻 Feed流就是不再是用户自己找 而是服务端主动投喂他喜欢/想看到的信息 考虑以下场景: 张三关注了李四 王五关注了李四 当李四发了动态时 它的粉丝们在我的关注列表里就能看到自己关注的人发的动态且最…

详解CountDownLatch底层源码

大家好&#xff0c;我是此林。 今天来分享一下CountDownLatch的底层源码。 CountDownLatch 是 Java 并发包 (java.util.concurrent) 中的线程之间同步工具类&#xff0c;主要用于协调多个线程的执行顺序。其核心思想是通过计数器实现线程间的"等待-唤醒"机制&#…

Ubuntu24.04 离线安装 MySQL8.0.41

一、环境准备 1.1 官方下载MySQL8.0.41 完整包 1.2 上传包 & 解压 上传包名称是&#xff1a;mysql-server_8.0.41-1ubuntu24.04_amd64.deb-bundle.tar # 切换到上传目录 cd /home/MySQL8 # 解压&#xff1a; tar -xvf mysql-server_8.0.41-1ubuntu24.04_amd64.deb-bundl…

【算法应用】基于粒子群算法PSO求解无人机集群路径规划问题

目录 1.无人机路径规划模型2.粒子群算法PSO原理3.结果展示4.参考文献5.代码获取 1.无人机路径规划模型 路径最优性 为了实现UAV的高效运行&#xff0c;计划的路径需要在某一特定标准上达到最优。UAV飞行路径Xi表示为UAV需要飞过的一系列n个航路点&#xff0c;每个航路点对应搜…

电脑ip地址每次开机会换吗?全面解析

在探讨“电脑IP地址每次开机会换吗”这一问题时&#xff0c;我们首先需要明确的是&#xff0c;IP地址的更换情况并非一成不变&#xff0c;而是受到多种因素的影响&#xff0c;其中最核心的是IP地址的类型——动态IP还是静态IP。这两种类型的IP地址在分配方式、稳定性以及使用场…

sqli-labs靶场 less 8

文章目录 sqli-labs靶场less 8 布尔盲注 sqli-labs靶场 每道题都从以下模板讲解&#xff0c;并且每个步骤都有图片&#xff0c;清晰明了&#xff0c;便于复盘。 sql注入的基本步骤 注入点注入类型 字符型&#xff1a;判断闭合方式 &#xff08;‘、"、’、“”&#xf…

docker-Dify外接Fastgpt知识库

参考地址&#xff1a;https://mp.weixin.qq.com/s/crQrneHZ0sT-c04YanofSw 总体步骤 部署fda(fastgpt-dify-adapter)docker 部署dify&#xff0c;fastgpt在fastgpt创建open apikey&#xff0c;复制知识库id&#xff1b;在dify外接fastgpt知识库&#xff1b; docker安装 下载…

Django学习笔记

Django学习笔记 安装django pip install django创建APP 用django来写后端的时候&#xff0c;要把各个功能分散到各个创建好的APP去实现 在终端输入 python manage.py startapp app01(APP名称)APP内部文件 admin.py django默认提供了admin后台管理 apps.py app启动类 mo…

向量数据库是什么,它有什么作用?

环境&#xff1a; 向量数据库 问题描述&#xff1a; 向量数据库是什么&#xff0c;它有什么作用 解决方案&#xff1a; 向量数据库是一种专门设计用于高效处理高维向量数据的系统&#xff0c;主要用于存储、索引、查询和检索高维向量数据&#xff0c;特别适合处理非结构化数…

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

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

【Portainer】Docker可视化组件安装

Portainer Portainer 是用于管理容器化环境的一体化平台工程解决方案&#xff0c;提供广泛的定制功能&#xff0c;以满足个人开发人员和企业团队的需求。 官方地址: https://www.portainer.io/ 安装 在 WSL / Docker Desktop 上使用 Docker 安装 Portainer CE 通过命令或UI页…

【第33节】windows原理:初探PE文件

目录 一、PE文件概述 二、DOS头部 三、DOS头部与NT头部之间 四、NT头部 五、文件头区段 六、了解个别概念 七、扩展头 八、区段头表 一、PE文件概述 PE文件是有特定格式的文件&#xff0c;像后缀名是EXE的可执行文件、后缀名是DLL的动态链接库文件、sys格式的驱动文件&…

谷粒微服务高级篇学习笔记整理---异步线程池

多线程回顾 多线程实现的4种方式 1. 继承 Thread 类 通过继承 Thread 类并重写 run() 方法实现多线程。 public class MyThread extends Thread {Overridepublic void run() {System.out.println("线程运行: " Thread.currentThread().getName());} }// 使用 pub…

网络运维学习笔记(DeepSeek优化版) 024 HCIP-Datacom OSPF域内路由计算

文章目录 OSPF域内路由计算&#xff1a;单区域的路由计算一、OSPF单区域路由计算原理二、1类LSA详解2.1 1类LSA的作用与结构2.2 1类LSA的四种链路类型 三、OSPF路由表生成验证3.1 查看LSDB3.2 查看OSPF路由表3.3 查看全局路由表 四、2类LSA详解4.1 2类LSA的作用与生成条件4.2 2…

飞桨PP系列新成员PP-DocLayout开源,版面检测加速大模型数据构建,超百页文档图像一秒搞定

背景介绍 文档版面区域检测技术通过精准识别并定位文档中的标题、文本块、表格等元素及其空间布局关系&#xff0c;为后续文本分析构建结构化上下文&#xff0c;是文档图像智能处理流程的核心前置环节。随着大语言模型、文档多模态及RAG&#xff08;检索增强生成&#xff09;等…

以科技赋能,炫我云渲染受邀参加中关村文化科技融合影视精品创作研讨会!

在文化与科技深度融合的时代浪潮下&#xff0c;影视创作行业经历着前所未有的变革。影视创作行业发展态势迅猛&#xff0c; 同时也面临着诸多挑战。为促进影视创作行业的创新发展&#xff0c;加强业内交流与合作&#xff0c; 3月25日下午&#xff0c;海淀区文化创意产业协会举办…

NFS挂载异常排查记录

互相PING服务器看是否通&#xff1b;在ubuntu下看下服务器是否正常运行。导出目录是否导出了。最后发现在挂载目录的地方目录路径和后面没有加空格。