【c++修行之路】智能指针

news2025/1/11 3:40:18

文章目录

  • 前言
  • 为什么用智能指针
  • 智能指针简单实现
    • unique_ptr
    • shared_ptr
  • 循环引用和weak_ptr的引入
    • 循环引用
    • weak_ptr
  • 定制删除器

前言

大家好久不见,今天来学习有关智能指针的内容~

为什么用智能指针

假如我们有如下场景:

double Div()
{
	int x, y;
	cin >> x >> y;

	if (y == 0)
		throw "div Exception cause div 0!!!";
	else
		return (double)x / (double)y;
}

int main()
{
	int* p1 = new int;
	int* p2 = new int;

	Div();

	delete p1;
	delete p2;
}

由于p1、p2、都需要释放,因此一旦在p2、p3出现了异常我们要手动释放前面的资源,这样的方式特别麻烦,这里还仅仅只是两个资源,一旦涉及更多new的资源会更麻烦,为了解决这个问题,c++引入了智能指针解决这个问题。

智能指针简单实现

一般而言智能指针要有三个问题:
1、利用对象的生命周期来控制程序资源,RAII的思想。
2、像指针一样使用。
3、考虑拷贝的问题。

unique_ptr

上面的例子中,如果p2、div出现了问题,前面的资源就无法释放,虽然可以通过重新抛出的方式来解决,但让代码可读性变得很差,同时也非常麻烦。使用智能指针可以解决这个问题,下面是一个智能指针的实例:

template<class T>
class SmartPtr
{
public:
	//构造保存指针
	SmartPtr(T* ptr)
		: _ptr(ptr)
	{

	}
	//析构释放资源
	~SmartPtr()
	{
		if(_ptr)
			delete _ptr;

		cout << "delete _ptr success !" << endl;
	}

	//模拟指针的两个行为
	T& operator*()
	{
		return *_ptr;
	}

	T* operator->()
	{
		return _ptr;
	}
private:
	T* _ptr;
};

这样完美地达到了控制资源释放的要求,但与此同时也引来了一个新的问题,即原生指针是可以拷贝的,但智能指针显然不可以拷贝,因为这里拷贝我们要求浅拷贝,那样对象在释放时同一份资源就会析构两次,这是非常可怕的。

为了解决这个问题,c++98库在实现auto_ptr的时候,使用了一种叫管理权转移的方式,如下代码,但其他这样效果非常不好。

//拷贝的悬空
SmartPtr(SmartPtr& sp)
	: _ptr(sp._ptr)
{
	sp._ptr = nullptr;
}

在后来的c++准标准库中,boost设计了一种新的智能指针,在c++11中相当于unique_ptr,该指针直接明令禁止不允许拷贝。

禁止别人拷贝的方式有很多,这里介绍两种:
在c++98中,一般只声明不实现,为了防止有人在类外动手脚,需要再用private封死这两个函数;相比之下c++11就更加简单,只需要写上delete关键字即可。

//c++11拷贝赋值禁止
	unique_ptr(const unique_ptr<T>& up) = delete;
	unique_ptr<T>& operator=(const unique_ptr<T>& up) = delete;
//c++98禁止:
private:
	unique_ptr(const unique_ptr<T>& up);
	unique_ptr<T>& operator=(const unique_ptr<T>& up);

shared_ptr

可以看出上面解决拷贝问题的方式本质就是规避了这个问题,shared_ptr不同,他允许我们进行拷贝构造。

要解决拷贝问题,实际上就是要控制好对象何时释放资源、释放几次的问题,我们发现引用计数很适合解决这个问题。

template<class T>
class shared_ptr
{
public:
	//构造保存指针
	shared_ptr(T* ptr)
		: _ptr(ptr)
		, _count(new int(1))
		, _pmtx(new mutex)
	{

	}
	//拷贝
	shared_ptr(const shared_ptr<T>& sp)
		: _ptr(sp._ptr)
	{
		if (_ptr != sp._ptr)
		{
			sp._count++;
		}
	}


	//析构释放资源
	~shared_ptr()
	{
		if ((--(*_count) == 0))
		{
			delete _ptr;
			delete _count;
			cout << "delete _ptr;" << " delete _count; " << endl;
		}
		cout << "delete _ptr success !" << endl;
	}

	//模拟指针的两个行为
	T& operator*()
	{
		return *_ptr;
	}

	T* operator->()
	{
		return _ptr;
	}


private:
	T* _ptr = nullptr;
	int* _count;
	mutex* _pmtx;
};


上述代码简单模拟了shared_ptr,但我们发现一旦使用了引用计数,不可避免地会出现线程安全问题,为了解决线程安全的问题,我们要对这些地方加锁。

template<class T>
class shared_ptr
{
public:
	//构造保存指针
	shared_ptr(T* ptr=nullptr)
		: _ptr(ptr)
		, _count(new int(1))
		, _pmtx(new mutex)
	{

	}

	//	+
	void AddCount()
	{
		_pmtx->lock();
		++(*_count);
		_pmtx->unlock();
	}

	//	-
	void Release()
	{
		_pmtx->lock();
		bool deleteFlag = false;

		if (--(*_count) == 0)
		{
			delete _ptr;
			delete _count;
			cout << "delete " << _ptr << endl;
			deleteFlag = true;
		}

		_pmtx->unlock();
		
		if (deleteFlag)
			delete _pmtx;
	}


	//拷贝构造
	shared_ptr(const shared_ptr<T>& sp)
		: _ptr(sp._ptr)
		, _count(sp._count)
		, _pmtx(sp._pmtx)
	{
		AddCount();
	}


	//赋值
	shared_ptr<T>& operator=(const shared_ptr<T>& sp)
	{
		if (_ptr != sp._ptr)
		{
			//这里是赋值之前的-
			Release();

			_ptr = sp._ptr;
			_count = sp._count;
			_pmtx = sp._pmtx;
			//注意这里加就是赋值之后的加了
			AddCount();
		}

		return *this;
	}


	//析构释放资源
	~shared_ptr()
	{
		Release();
	}

	//模拟指针的两个行为
	T& operator*()
	{
		return *_ptr;
	}

	T* operator->()
	{
		return _ptr;
	}

	T* get() const
	{
		return _ptr;
	}

	int use_count()
	{
		return *_count;
	}

private:
	T* _ptr;
	int* _count;
	mutex* _pmtx;
};

注意,shared_ptr本身是线程安全的,但管理的对象并不是线程安全的,需要加锁保护,在一些极端的场景下还会出现循环引用的问题。

循环引用和weak_ptr的引入

循环引用

struct ListNode
{
	int _val;
	nhy::shared_ptr<ListNode> _prev;
	nhy::shared_ptr<ListNode> _next;

	~ListNode()
	{
		cout << "~ListNode" << endl;
	}
};

如果链表的两个指针也使用shared_ptr,就会出现如图所示循环引用的问题,不仅仅p指向这个对象,还有一个next或prev也指向这个对象,这样双方会僵持不下,谁都无法释放。
在这里插入图片描述

weak_ptr

要解决这个问题,只需要将不必要的计数功能取消即可,其实weakptr本质就是sharedptr取消了计数功能。

template<class T>
class weak_ptr
{
public:
	weak_ptr()
		:_ptr(nullptr)
	{}

	weak_ptr(const shared_ptr<T>& sp)
		:_ptr(sp.get())
	{}

	T& operator*()
	{
		return *_ptr;
	}

	T* operator->()
	{
		return _ptr;
	}

	T* get()
	{
		return _ptr;
	}

private:
	T* _ptr;
};

定制删除器

可以传入一个对象来管理释放资源。

template<class D>
shared_ptr(T* ptr, D del)
	: _ptr(ptr)
	, _count(new int(1))
	, _pmtx(new mutex)
	, _del(del)
{

}

function<void(T*)> _del = [](T* ptr) {
	cout << "default delete" << endl;
	delete ptr;
};

//  定制删除器 -- 可调用对象
template<class T>
struct DeleteArray
{
	void operator()(T* ptr)
	{
		cout << "void operator()(T* ptr)" << endl;
		delete[] ptr;
	}
};


class Date
{
private:
	int _year;
	int _month;
	int _day;
};

void test_delete()
{
	 nhy::shared_ptr<int> spa1(new int[10],DeleteArray<int>());
	 nhy::shared_ptr<Date> spa2(new Date[10]);
	 nhy::shared_ptr<Date> spa3(new Date[10],DeleteArray<Date>());
}

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

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

相关文章

Clion 配置Mingw64的 c++开发环境

1、Mingw64的安装与环境变量的配置 Mingw64文件下载 Mingw64下载地址&#xff1a;https://sourceforge.net/projects/mingw-w64/files/ posix相比win32拥有C 11多线程特性&#xff0c;sjlj和seh对应异常处理特性&#xff0c;sjlj较为古老&#xff0c;所以选择seh 配置环境变…

MongoDB踩过的坑

目录 启动MongoDB服务 可视化工具&#xff1a;MongoDB Compass 由于目标计算机积极拒绝&#xff0c;无法连接 BSONObj size: xxxx is invalid. Size must be between 0 and 16793600 (16MB) 启动MongoDB服务 1. 打开CMD 2. 进入安装MongoDB文件夹中的bin目录 3. mongod -…

mapBox 绘制多边形无法设置 边框宽度 解决方法

目录 一、问题 二、解决方法 三、总结 tips:如嫌繁琐&#xff0c;直接看有颜色的文字即可&#xff01; 一、问题 1.使用mapBox在地图上绘制点、线、面。绘制多边形的时候发现 直接用 zh(一家提供地图引擎的公司),提供的绘制多边形的方法无法设置边框颜色和边框宽度。很是离…

龙蜥社区 6 月技术委员会会议召开!欢迎 5 位开放原子 TOC 导师加入

2023 年 6 月 16 日上午 10 点召开了龙蜥社区 6 月技术委员会线上会议&#xff0c;共计 38 人参会。本次会议由联通肖微主持&#xff0c;会议也荣幸的邀请到了开放原子 TOC 导师线上参会&#xff0c;技术委员们来自阿里云、统信、飞腾、中科方德、红旗、万里红、Intel、Arm、龙…

入门篇:从零上手GitOps

文章目录 GitOps 介绍如何将业务代码构建为容器镜像&#xff1f;如何将容器镜像部署到K8s&#xff1f;K8s如何实现自动扩容和自愈&#xff1f;1.传统的扩容和自愈2.k8s自愈机制3.k8s弹性扩容 如何借助GitOps实现应用秒级自动发布和回滚&#xff1f;1.传统 K8s 应用发布流程2.从…

高级细腻的家居照明,欧瑞博智能无主灯是怎么实现的?

作者 | 辰纹 来源 | 洞见新研社 如今的现代生活&#xff0c;人类对光的需求已超越简单照明&#xff0c;而是希望在不同场景下能有专属的细腻用光体验&#xff0c;智能照明应运而生&#xff0c;并成为上升趋势。现阶段&#xff0c;精细化家居需求要求智能照明不仅要巧妙融合美学…

二叉树进阶(AVLTree)

目录 1.AVLTree概念 2.AVLTree模拟实现 2.1 AVLTree节点 2.2 插入实现基本框架 2.3 左单旋 2.4 右单旋 2.5 LR双旋 2.6 RL双旋 2.7 AVLTree树验证 1.AVLTree概念 二叉搜索树虽可以缩短查找的效率&#xff0c;但如果数据有序或接近有序二叉搜索树将退化为单支树&#…

怎么从电影中截取动图?试试这个工具

图片、视频等都是当代流行的表达情感、传递信息的一种方式。其中&#xff0c;当属gif动图最受大众的欢迎&#xff0c;它比普通的静态图片画面丰富&#xff0c;又比视频的体积小。那么&#xff0c;如何从视频中截取动图呢&#xff1f;使用GIF中文网的视频转gif&#xff08;https…

通过platform实现阻塞IO来驱动按键控制LED灯的亮灭

通过platform阻塞IO来驱动按键控制LED灯的亮灭 a .应用程序通过阻塞的io模型来读取number变量的值 b.number是内核驱动中的一个变量 c .number的值随着按键按下而改变&#xff08;按键中断)例如number0按下按键number1 ,再次按下按键number0 d .在按下按键的时候需要同时将…

【Leetcode】42.接雨水(困难)

一、题目 1、题目描述 给定 n 个非负整数表示每个宽度为 1 的柱子的高度图,计算按此排列的柱子,下雨之后能接多少雨水。 示例1: 输入:height = [0,1,0,2,1,0,1,3,2,1,2,1] 输出:6 解释:上面是由数组 [0,1,0,2,1,0,1,3,2,1,2,1] 表示的高度图,在这种情况下,可以接 6…

雪花算法 — 集群高并发情况下如何保证分布式唯一全局ID生成?

雪花算法 问题 为什么需要分布式全局唯一ID以及分布式ID的业务需求 在复杂分布式系统中&#xff0c;往往需要对大量的数据和消息进行唯一标识&#xff1a; 如在美团点评的金融、支付、餐饮、酒店猫眼电影等产品的系统中数据逐渐增长&#xff0c;对数据库分库分表后需要有一…

接口测试辅助,Fiddler抓取安卓手机https请求(详细)

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 Fiddler 是一款免…

Java设计模式之行为型-迭代器模式(UML类图+案例分析)

目录 一、基础概念 二、UML类图 三、角色设计 四、案例分析 五、总结 一、基础概念 迭代器模式是一种常用的设计模式&#xff0c;它主要用于遍历集合对象&#xff0c;提供一种方法顺序访问一个聚合对象中的各个元素&#xff0c;而又不暴露该对象的内部表示。 举个简单的…

第二章:Cyber RT通信机制解析与实践

Cyber RT解析与实践 第二章&#xff1a;Cyber RT通信机制解析与实践 Cyber RT解析与实践 Cyber RT解析与实践一、Cyber RT 通讯机制简介1. 话题2. 服务3. 参数 二、数据通信基础Protobuf1. Protobuf 简介2. Protobuf 创建3. Protobuf 编译4. Protobuf 案例实战 三、Cyber RT 话…

CPU性能指标简览

作为计算机的运算核心和控制核心&#xff0c;CPU&#xff08;Central Processing Unit&#xff09;由运算器、控制器、寄存器和实现其之间联系的数据、控制及状态的总线构成&#xff0c;决定着计算机运算性能强弱。作为信息技术产业的核心基础元器件&#xff0c;CPU的运作可分为…

3-40V输入,2.7V启动,20A电流,PWM\模拟信号调光

应用说明&#xff1a; Hi600X 是一系列外围电路简洁的宽调光比升压恒流驱动器&#xff0c;适用于 3-40V 输入电压范围的 LED 照明领域。 Hi600X 系列芯片&#xff0c;2.7V 启动电压&#xff0c;工作电压范围 5-40V&#xff0c;VIFB反馈电压 0.2V&#xff0c;提高整体转换效率。…

JVM的类加载机制和垃圾回收机制

目录 类加载机制类加载机制的步骤加载验证准备解析初始化 双亲委派模型工作原理双亲委派模型的优点 垃圾回收机制死亡对象的判断可达性分析算法可达性分析算法的缺点引用计数算法循环引用问题 垃圾回收算法标记-清除算法复制算法标记-整理算法分代算法 类加载机制 对于任意一个…

使用 vector + 递归 解决问题

class Solution {const char* To[10] { "","", "abc", "def", "ghi","jkl", "mno", "pqrs","tuv", "wxyz" };//常量字符串初始化string 注意此非定义(缺省值)--实例化…

收好这5个SVG编辑器,轻松实现高效创作

随着设计工作的不断发展&#xff0c;对SVG图形的需求也逐渐增加&#xff0c;SVG编辑器就随之诞生。可市面上的SVG编辑质量参差不齐&#xff0c;设计师无法第一时间找到适合自己的&#xff0c;于是本文就收集整理了5款相对来说比较好用的SVG编辑器为设计师们推荐&#xff0c;一起…

java项目之个人网站(ssm+mysql+jsp)

风定落花生&#xff0c;歌声逐流水&#xff0c;大家好我是风歌&#xff0c;混迹在java圈的辛苦码农。今天要和大家聊的是一款基于ssm的个人网站。技术交流和部署相关看文章末尾&#xff01; 开发环境&#xff1a; 后端&#xff1a; 开发语言&#xff1a;Java 框架&#xf…