第7章 Page449~451 7.8.9智能指针 std::shared_ptr

2024-02-17 16:04

本文主要是介绍第7章 Page449~451 7.8.9智能指针 std::shared_ptr,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

“shared_ptr”是“共享式智能指针”。

即多个“shared_ptr”之间可以管理同一个裸指针。于是

O* o = new O; //一个裸指针
std::shared_ptr <O> p1(o); //交给p1管
std::shared_ptr <O> p2(o); //又交给p2管

出乎意料,以上代码仍然是可以通过编译但运行期将出错。“一女二嫁”永远是错误的,并且永远是编译器所不能检测的。

演示:“一女二嫁”

 对比“unique_ptr”和“shared_ptr”,前者(独占式)认为女子可以改嫁,但应保持一夫一妻制;而后者(共享式)比较可怕,看实际代码

O* o = new o;//还是一个裸指针
std::shared_ptr <O> p1(o);//先交给p1
std::shared_ptr <O> p2 = p1;

003行的结果,是让p2和p1同指向。严格来讲是管理同一个裸指针,那么问题来了,p2和p1同时管理同一个裸指针,接着p1结束生命,自动释放该裸指针,再接着p2也结束生命,于是再次释放该裸指针,不就造成一个指针被释放两次的结果吗?不会的。“std::shared_ptr”采用了相当复杂的技术来保证,当存在多个智能指针共同管理某一裸指针,仅当最后一个智能指针结束生命时,才会真正释放所共享的裸指针。

具体方法称为“计数法”。将共享式智能指针A赋值给共享式智能指针B,B将与A管理同一个裸指针,并且在系统某处记录,当前有两个智能指针在管理某一裸指针,简称“计数为二”。而当B或A退出(结束生命周期)时,该记数减为一,直到另一个也退出,计数归零,此时才真正释放裸指针。

演示:多个shared_ptr共同管理一个裸指针

“std::shared_ptr”的使用方法和具体功能:

初始化:

std::shared_ptr <O> p1(new O());

也可以采用“列表式初始化”语法,即花括号:

std::shared_ptr <O> p1{new O()};

但是同样不能使用“=”将一个裸指针“赋值”给另一个智能指针:

std::shared_ptr <O> p1 = new O(); //ERROR

构造时没有提供裸指针,得到空指向的智能指针:

std::shared_ptr <O> p1; //p1是空指向

同样对nullptr做了特殊处理,允许直接赋值为nullptr;

p1 = nullptr; //p1变成空指向,原管理的裸指针被释放

或者调用无参版的“reset()”:

p1.reset(); //p1变成空指向,原管理的裸指针被释放

和“unique_ptr”一样,同样可以通过带参版“reset(...)”改变指向,代码略。

标准库还提供了“make_shared()”模板函数,用于更加高效地创建“shared_ptr”:

std::shard_ptr <O> p1 = std::make_shared <O> ();

如果所要构建的对象需要入参,则通过“make_shared()”函数传递:

std::shared_ptr <std::string> ps= std::make_shared <std::string> ("Hello!");
cout << *ps << endl;

演示:make_shared()

课堂作业:正确令多个“shared_ptr”共享管理同一裸指针

多个裸指针可以指向同一份数据,因此可以将裸指针划分成“共享式”,只是裸指针无法自动释放。因为同属“共享式”,所以“std::share_ptr”使用起来更接近裸指针。比如裸指针和“shared_ptr”都支持加入到容器中管理,“unique_ptr”却“做不到”:

unique_ptr <int> p(new int);
list <unique_ptr<int>> l;
l.push_back(p);

编译至002行仍未出错,但一旦真要将某个独占式智能指针,加入到容器中,就会报错。

因为独占式指针对象不允许复制(只允许转移),而容器要存储、管理元素,躲不过复制操作。

【重要】杀鸡用什么刀

为什么有功能丰富的“shared_ptr”,还要有“unique_ptr”呢?原因在于存在大量无需“共享指向”的指针应用场景。

很多时候功能越丰富,越容易在使用上出错。裸指针本身就是个案例。相比裸指针,“unique_ptr”往“简化、易用”的方向设计。另一方面,为了可以共享指向,“shared_ptr”需要付出性能代价。

程序员必须习惯于做预分析,能使用“unique_ptr”解决问题就使用“unique_ptr”,莫因图强大和适用面广而上来就用“shared_ptr”。

这篇关于第7章 Page449~451 7.8.9智能指针 std::shared_ptr的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/718288

相关文章

嵌入式QT开发:构建高效智能的嵌入式系统

摘要: 本文深入探讨了嵌入式 QT 相关的各个方面。从 QT 框架的基础架构和核心概念出发,详细阐述了其在嵌入式环境中的优势与特点。文中分析了嵌入式 QT 的开发环境搭建过程,包括交叉编译工具链的配置等关键步骤。进一步探讨了嵌入式 QT 的界面设计与开发,涵盖了从基本控件的使用到复杂界面布局的构建。同时也深入研究了信号与槽机制在嵌入式系统中的应用,以及嵌入式 QT 与硬件设备的交互,包括输入输出设

让树莓派智能语音助手实现定时提醒功能

最初的时候是想直接在rasa 的chatbot上实现,因为rasa本身是带有remindschedule模块的。不过经过一番折腾后,忽然发现,chatbot上实现的定时,语音助手不一定会有响应。因为,我目前语音助手的代码设置了长时间无应答会结束对话,这样一来,chatbot定时提醒的触发就不会被语音助手获悉。那怎么让语音助手也具有定时提醒功能呢? 我最后选择的方法是用threading.Time

智能交通(二)——Spinger特刊推荐

特刊征稿 01  期刊名称: Autonomous Intelligent Systems  特刊名称: Understanding the Policy Shift  with the Digital Twins in Smart  Transportation and Mobility 截止时间: 开放提交:2024年1月20日 提交截止日

基于 YOLOv5 的积水检测系统:打造高效智能的智慧城市应用

在城市发展中,积水问题日益严重,特别是在大雨过后,积水往往会影响交通甚至威胁人们的安全。通过现代计算机视觉技术,我们能够智能化地检测和识别积水区域,减少潜在危险。本文将介绍如何使用 YOLOv5 和 PyQt5 搭建一个积水检测系统,结合深度学习和直观的图形界面,为用户提供高效的解决方案。 源码地址: PyQt5+YoloV5 实现积水检测系统 预览: 项目背景

【C++学习笔记 20】C++中的智能指针

智能指针的功能 在上一篇笔记提到了在栈和堆上创建变量的区别,使用new关键字创建变量时,需要搭配delete关键字销毁变量。而智能指针的作用就是调用new分配内存时,不必自己去调用delete,甚至不用调用new。 智能指针实际上就是对原始指针的包装。 unique_ptr 最简单的智能指针,是一种作用域指针,意思是当指针超出该作用域时,会自动调用delete。它名为unique的原因是这个

C语言指针入门 《C语言非常道》

C语言指针入门 《C语言非常道》 作为一个程序员,我接触 C 语言有十年了。有的朋友让我推荐 C 语言的参考书,我不敢乱推荐,尤其是国内作者写的书,往往七拼八凑,漏洞百出。 但是,李忠老师的《C语言非常道》值得一读。对了,李老师有个官网,网址是: 李忠老师官网 最棒的是,有配套的教学视频,可以试看。 试看点这里 接下来言归正传,讲解指针。以下内容很多都参考了李忠老师的《C语言非

单片机毕业设计基于单片机的智能门禁系统的设计与实现

文章目录 前言资料获取设计介绍功能介绍程序代码部分参考 设计清单具体实现截图参考文献设计获取 前言 💗博主介绍:✌全网粉丝10W+,CSDN特邀作者、博客专家、CSDN新星计划导师,一名热衷于单片机技术探索与分享的博主、专注于 精通51/STM32/MSP430/AVR等单片机设计 主要对象是咱们电子相关专业的大学生,希望您们都共创辉煌!✌💗 👇🏻 精彩专栏 推荐订

C和指针:字符串

字符串、字符和字节 字符串基础 字符串就是一串零个或多个字符,并且以一个位模式为全0的NUL字节结尾。 字符串长度就是字符串中字符数。 size_t strlen( char const *string ); string为指针常量(const修饰string),指向的string是常量不能修改。size_t是无符号数,定义在stddef.h。 #include <stddef.h>

【C++】作用域指针、智能指针、共享指针、弱指针

十、智能指针、共享指针 从上篇文章 【C++】如何用C++创建对象,理解作用域、堆栈、内存分配-CSDN博客 中我们知道,你的对象是创建在栈上还是在堆上,最大的区别就是对象的作用域不一样。所以在C++中,一旦程序进入另外一个作用域,那其他作用域的对象就自动销毁了。这种机制有好有坏。我们可以利用这个机制,比如可以自动化我们的代码,像智能指针、作用域锁(scoped_lock)等都是利用了这种机制。

MFC中App,Doc,MainFrame,View各指针的互相获取

纸上得来终觉浅,为了熟悉获取方法,我建了个SDI。 首先说明这四个类的执行顺序是App->Doc->Main->View 另外添加CDialog类获得各个指针的方法。 多文档的获取有点小区别,有时间也总结一下。 //  App void CSDIApp::OnApp() {      //  App      //  Doc     CDocument *pD