首页 > 其他分享 >cpp智能指针

cpp智能指针

时间:2024-12-20 16:56:45浏览次数:5  
标签:AA 智能 new cpp shared unique ptr 指针

  普通指针的不足

  • new和new[]的内存需要用delete和deletel]释放。

  • 程序员的主观失误,忘了或漏了释放。

  • 程序员也不确定何时释放。

普通指针的释放

  • 类内的指针,在析构函数中释放。

  • C++内置数据类型,如何释放?

  • new出来的类,本身如何释放?

C++11新增三个智能指针类型

  • unique_ptr

  • shared_ptr

  • weak_ptr

一、智能指针unique_ptr

​        unique_ptr独享它指向的对象,也就是说,同时只有一个unique_ptr指向同一个对象,当这个unique_ptr被销毁时,指向的对象也随即被销毁。unique_ptr是一个类,它的内部维护一个内置的指针,他的析构函数中会delete内置指针。

包含头文件:#include <memory>

template <typename T, typename D = default_delete<T>>
class unique_ptr
{
public:
	explicit unique_ptr(pointer p) noexcept;	// 不可用于转换函数。
	~unique_ptr() noexcept;    
	T& operator*() const;            // 重载*操作符。
	T* operator->() const noexcept;  // 重载->操作符。
	unique_ptr(const unique_ptr &) = delete;   // 禁用拷贝构造函数。
	unique_ptr& operator=(const unique_ptr &) = delete;  // 禁用赋值函数。
	unique_ptr(unique_ptr &&) noexcept;	  // 右值引用。
	unique_ptr& operator=(unique_ptr &&) noexcept;  // 右值引用。
	// ...
private:
	pointer ptr;  // 内置的指针。
};

explicit 关键字

加上explicit后让构造函数不可以被用作转换函数使用。以下调用非法:

AA *p = new AA("xianwu");	//普通指针
unique_ptr<AA> pu2 = p;	

1、基本用法

1)初始化

方法一:推荐

unique_ptr<AA> p0(new AA("西施"));   // 分配内存并初始化。

方法二:

unique_ptr<AA> p0 = make_unique<AA>("西施");  // C++14标准。

方法三(不推荐):原始指针出现如果构造给多个unique_ptr会造成多次释放内存,操作野指针,存在风险。

AA* p = new AA("西施");

unique_ptr<AA> p0(p);         // 用已存在的地址初始化。

2)使用方法

  • 智能指针重载了*和->操作符,可以像使用指针一样使用unique_ptr。
  • 不支持普通的拷贝和赋值。
AA* p = new AA("西施");
unique_ptr<AA> pu2 = p;       // 错误,不能把普通指针直接赋给智能指针。
unique_ptr<AA> pu3 = new AA("西施"); // 错误,不能把普通指针直接赋给智能指针。
unique_ptr<AA> pu2 = pu1;      // 错误,不能用其它unique_ptr拷贝构造。
unique_ptr<AA> pu3;
pu3 = pu1;              // 错误,不能用=对unique_ptr进行赋值。
  • 不要用同一个裸指针初始化多个unique_ptr对象。
  • get()方法返回裸指针。
  • 不要用unique_ptr管理不是new分配的内存。

3)用于函数的参数

  • 传引用(不能传值,因为unique_ptr没有拷贝构造函数)。

  • 裸指针。

4)不支持指针的运算(+、-、++、--)

2、更多技巧 面试就爱考点技巧之类的

1)将一个unique_ptr赋给另一个时,如果源unique_ptr是一个临时右值,编译器允许这样做;如果源unique_ptr将存在一段时间,编译器禁止这样做。一般用于函数的返回值。

unique_ptr<AA> p0;
p0 = unique_ptr<AA>(new AA ("西瓜"));

2)用nullptr给unique_ptr赋值将释放对象,空的unique_ptr==nullptr。

3)release()释放对原始指针的控制权,将unique_ptr置为空,返回裸指针。(可用于把unique_ptr传递给子函数,子函数将负责释放对象)

4)std::move()可以转移对原始指针的控制权。(可用于把unique_ptr传递给子函数,子函数形参也是unique_ptr)

5)reset()释放对象。

void reset(T * _ptr= (T *) nullptr);

pp.reset();    // 释放pp对象指向的资源对象。
pp.reset(nullptr); // 释放pp对象指向的资源对象
pp.reset(new AA("bbb")); // 释放pp指向的资源对象,同时指向新的对象。

6)swap()交换两个unique_ptr的控制权。

void swap(unique_ptr<T> &_Right);

7)unique_ptr也可像普通指针那样,当指向一个类继承体系的基类对象时,也具有多态性质,如同使用裸指针管理基类对象和派生类对象那样。

8)unique_ptr不是绝对安全,如果程序中调用exit()退出,全局的unique_ptr可以自动释放,但局部的unique_ptr无法释放。

9)unique_ptr提供了支持数组的具体化版本。

数组版本的unique_ptr,重载了操作符[],操作符[]返回的是引用,可以作为左值使用。

// unique_ptr<int[]> parr1(new int[3]);     // 不指定初始值。
unique_ptr<int[]> parr1(new int[3]{ 33,22,11 }); // 指定初始值。
cout << "parr1[0]=" << parr1[0] << endl;
cout << "parr1[1]=" << parr1[1] << endl;
cout << "parr1[2]=" << parr1[2] << endl;
 
unique_ptr<AA[]> parr2(new AA[3]{string("西施"), string("冰冰"), string("幂幂")});
cout << "parr2[0].m_name=" << parr2[0].m_name << endl;
cout << "parr2[1].m_name=" << parr2[1].m_name << endl;
cout << "parr2[2].m_name=" << parr2[2].m_name << endl;

 

二、智能指针shared_ptr

shared_ptr共享它指向的对象,多个shared_ptr可以指向(关联)相同的对象,在内部采用计数机制来实现,use_count()。

当新的shared_ptr与对象关联时,引用计数增加1。

当shared_ptr超出作用域时,引用计数减1。当引用计数变为0时,则表示没有任何shared_ptr与对象关联,则释放该对象。

 1、基本用法

​        shared_ptr的构造函数也是explicit,但是,没有删除拷贝构造函数和赋值函数。

1)初始化

方法一:

shared_ptr<AA> p0(new AA("西施"));   // 分配内存并初始化。

方法二:推荐采用

shared_ptr<AA> p0 = make_shared<AA>("西施"); // C++11标准,效率更高。

方法三:

AA* p = new AA("西施");
shared_ptr<AA> p0(p);         // 用已存在的地址初始化。

方法四:

shared_ptr<AA> p0(new AA("西施")); 
shared_ptr<AA> p1(p0);         // 用已存在的shared_ptr初始化,计数加1。
shared_ptr<AA> p1=p0;         // 用已存在的shared_ptr初始化,计数加1。

2)使用方法

  • 智能指针重载了*和->操作符,可以像使用指针一样使用shared_ptr。

  • use_count()方法返回引用计数器的值。

  • unique()方法,如果use_count()为1,返回true,否则返回false。

  • shared_ptr支持赋值,左值的shared_ptr的计数器将减1,右值shared_ptr的计算器将加1。

  • get()方法返回裸指针。

  • 不要用同一个裸指针初始化多个shared_ptr。即不要用方法三方式构造多个shared_ptr。

  • 不要用shared_ptr管理不是new分配的内存。

3)用于函数的参数

与unique_ptr的原理相同。

  • 传引用(不能传值,因为unique_ptr没有拷贝构造函数)。

  • 裸指针。

4)不支持指针的运算(+、-、++、--)

2、更多细节 面试就爱考点技巧之类的

2)用nullptr给shared_ptr赋值将把计数减1,如果计数为0,将释放对象,空的shared_ptr==nullptr。

4)std::move()可以转移对原始指针的控制权。还可以将unique_ptr转移成shared_ptr。

5)reset()改变与资源的关联关系。

pp.reset();     // 解除与资源的关系,资源的引用计数减1。
pp. reset(new AA("bbb"));  // 解除与资源的关系,资源的引用计数减1。关联新资源。

6)swap()交换两个shared_ptr的控制权。

void swap(shared_ptr<T> &_Right);

7)shared_ptr也可象普通指针那样,当指向一个类继承体系的基类对象时,也具有多态性质,如同使用裸指针管理基类对象和派生类对象那样。

8)shared_ptr不是绝对安全,如果程序中调用exit()退出,全局的shared_ptr可以自动释放,但局部的shared_ptr无法释放。

9)shared_ptr提供了支持数组的具体化版本。

数组版本的shared_ptr,重载了操作符[],操作符[]返回的是引用,可以作为左值使用。

10)shared_ptr的线程安全性:

  • shared_ptr的引用计数本身是线程安全(引用计数是原子操作)。

  • 多个线程同时读同一个shared_ptr对象是线程安全的。

  • 如果是多个线程对同一个shared_ptr对象进行读和写,则需要加锁。

  • 多线程读写shared_ptr所指向的同一个对象,不管是相同的shared_ptr对象,还是不同的shared_ptr对象,也需要加锁保护。

11)如果unique_ptr能解决问题,就不要用shared_ptr。unique_ptr的效率更高,占用的资源更少。

3、智能指针的删除器

智能指针构造函数第二个参数一般用默认的,但可以指定删除器。

在默认情况下,智能指针过期的时候,用delete原始指针;释放它管理的资源。

程序员可以自定义删除器,改变智能指针释放资源的行为。

删除器可以是全局函数、仿函数和Lambda表达式,形参为原始指针。

调用:

shared_ptr<AA> pa1(new AA("西施a"), deletefunc);  // 第二个参数一般都默认不写,但可以自定义删除器。

三、智能指针weak_ptr

1、shared_ptr存在的问题

shared_ptr内部维护了一个共享的引用计数器,多个shared_ptr可以指向同一个资源。

如果出现了循环引用的情况,引用计数永远无法归0,资源不会被释放。

2、weak_ptr是什么

weak_ptr 是为了配合shared_ptr而引入的,它指向一个由shared_ptr管理的资源但不影响资源的生命周期。也就是说,将一个weak_ptr绑定到一个shared_ptr不会改变shared_ptr的引用计数。

不论是否有weak_ptr指向,如果最后一个指向资源的shared_ptr被销毁,资源就会被释放。

weak_ptr更像是shared_ptr的助手而不是智能指针。

3、如何使用weak_ptr

weak_ptr没有重载 ->和 *操作符,不能直接访问资源。

有以下成员函数:

operator=(); // 把shared_ptr或weak_ptr赋值给weak_ptr。
expired();   // 判断它指资源是否已过期(已经被销毁)。
lock();    // 返回shared_ptr,如果资源已过期,返回空的shared_ptr。
reset();    // 将当前weak_ptr指针置为空。
swap();    // 交换。

weak_ptr不控制对象的生命周期,但是,它知道对象是否还活着。

用lock()函数把它可以提升为shared_ptr,如果对象还活着,返回有效的shared_ptr,如果对象已经死了,提升会失败,返回一个空的shared_ptr。

提升的行为(lock())是线程安全的。

标签:AA,智能,new,cpp,shared,unique,ptr,指针
From: https://blog.csdn.net/weixin_56520780/article/details/144602228

相关文章

  • 智能脂肪秤方案app蓝牙称重方案研发
    智能脂肪秤方案,一种测量仪器,针对人体体重智能分析的仪器,相比传统的指针式体重秤,具有精度高、锁定显示、读数方便、体积小、重量轻等优势。这种智能脂肪秤,由于其测量的数据精准和价格便宜,面向的也不仅仅是医疗行业,更多的是面对个人家庭,因为,现如今由于人们对于消费观念的改变,导......
  • 最强AI智能体竟如此简单!Anthropic首次公开内部秘籍!
    0前言过去一年中,我们与不同行业中开发大语言模型(LLM)智能体的多个团队进行了合作。我们发现,最成功的实现并非依赖复杂的框架或专业化的库,而是通过简单、可组合的模式构建。本文分享从客户合作及自身开发智能体的过程中所学到的经验,并为开发者提供构建高效智能体的实用建议。......
  • Agent智能体,打响中国大模型落地产业第一枪
    与大模型不同,Agent就像是一个“标准答案”,直接摆在企业面前,就连使用说明书和企业最看重的投入回报比都计算好了。前排提示,文末有大模型AGI-CSDN独家资料包哦!“今年,我们是针对具体的场景构建大模型,等到明年会进行更深一步的探索。目前我们正在和云服务伙伴讨论构建数字员工......
  • AI智能分析视频分析网关关于监控摄像头数据量对云存储服务影响的探讨
    在数字化安防监控领域,监控摄像头数据量的激增对云存储服务带来了前所未有的挑战与机遇。随着技术的不断进步和监控需求的日益增长,监控摄像头所产生的数据量正以前所未有的速度增长,这对云存储服务的性能、容量及可靠性提出了更高要求。本文将深入探讨监控摄像头数据量对云存储服务......
  • DDoS防护中的流量清洗与智能调度
    DDoS防护中的流量清洗与智能调度有哪些好处在数字化高度发展的今天,企业依赖于互联网进行业务运营,而网络安全威胁也随之增加。其中,DDoS(分布式拒绝服务)攻击是一种常见且破坏性极强的网络攻击手段。为了有效应对DDoS攻击,流量清洗和智能调度已成为现代DDoS防护的重要技术。本文......
  • 面向教学科研智能感知系统应用开发实验室
    概述    车辆感知系统在智能驾驶中扮演着至关重要的角色,它如同车辆的“眼睛”和“耳朵”,负责实时监测和解读周围环境信息。该系统通过集成摄像头、雷达、激光雷达等多种传感器,能够准确识别道路状况、行人和其他车辆,为智能驾驶提供可靠的数据支持。经纬恒润推出面向教学及......
  • 角球预测分析方法推荐:AI大数据智能预测
    一、引言在竞技体育领域,预测比赛中的关键事件一直是数据分析的热点。本文旨在为专业的编程师和体育数据分析爱好者提供一种基于人工智能(AI)和大数据技术的角球预测分析方法。我们将探讨如何运用机器学习算法,结合时间序列分析,实现对比赛过程中角球事件的精准预测。二、数据预......
  • 留学界的 “智能力量”:思沃克 NexGPT 如何革新行业服务格局
    在竞争日益激烈的留学教育行业,各机构都在竭力探寻提升服务质量与效率的方法,以满足学生和家长们日益增长的需求和期望。而思沃克NexGPT大模型解决方案的出现,犹如一场及时雨,为整个留学教育行业带来了全新的活力与变革契机。思沃克NexGPT的核心在于其先进的大模型知识库和智能体......
  • Excel设置生日自动智能提醒,公式可直接套用!
    大家好,我是小鱼。    今天跟大家分享一个WPS表格中根据出生日期,设置生日提醒,并且根据距离生日天数自动标记数据颜色。简单又实用,一个公式轻松搞定!接下来我们先学习一下需要使用到的函数,然后再根据实例让大家彻底掌握必备的生日自动提醒技能。函数简介:1、TEXT函......
  • 助力现代农林业病虫害智能化监管治理,基于YOLOv5全系列【n/s/m/l/x】参数模型开发构建
    在传统的农林业管理中,病虫害的监管和治理往往依赖于经验丰富的师傅。他们根据树木的不同季节和不同时段,进行针对性的分析和治疗。然而,这种传统的工作模式存在诸多局限性,如无法实现24小时不间断工作,且随着种植规模的扩大,一些出现病害的树木难以被及时发现和有效治理。随着人工智......