第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

相关文章

Java空指针异常NullPointerException的原因与解决方案

《Java空指针异常NullPointerException的原因与解决方案》在Java开发中,NullPointerException(空指针异常)是最常见的运行时异常之一,通常发生在程序尝试访问或... 目录一、空指针异常产生的原因1. 变量未初始化2. 对象引用被显式置为null3. 方法返回null

基于Python实现智能天气提醒助手

《基于Python实现智能天气提醒助手》这篇文章主要来和大家分享一个实用的Python天气提醒助手开发方案,这个工具可以方便地集成到青龙面板或其他调度框架中使用,有需要的小伙伴可以参考一下... 目录项目概述核心功能技术实现1. 天气API集成2. AI建议生成3. 消息推送环境配置使用方法完整代码项目特点

JavaScript实战:智能密码生成器开发指南

本文通过JavaScript实战开发智能密码生成器,详解如何运用crypto.getRandomValues实现加密级随机密码生成,包含多字符组合、安全强度可视化、易混淆字符排除等企业级功能。学习密码强度检测算法与信息熵计算原理,获取可直接嵌入项目的完整代码,提升Web应用的安全开发能力 目录

利用Python实现Excel文件智能合并工具

《利用Python实现Excel文件智能合并工具》有时候,我们需要将多个Excel文件按照特定顺序合并成一个文件,这样可以更方便地进行后续的数据处理和分析,下面我们看看如何使用Python实现Exce... 目录运行结果为什么需要这个工具技术实现工具的核心功能代码解析使用示例工具优化与扩展有时候,我们需要将

基于Python打造一个智能单词管理神器

《基于Python打造一个智能单词管理神器》这篇文章主要为大家详细介绍了如何使用Python打造一个智能单词管理神器,从查询到导出的一站式解决,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1. 项目概述:为什么需要这个工具2. 环境搭建与快速入门2.1 环境要求2.2 首次运行配置3. 核心功能使用指

Python实现word文档内容智能提取以及合成

《Python实现word文档内容智能提取以及合成》这篇文章主要为大家详细介绍了如何使用Python实现从10个左右的docx文档中抽取内容,再调整语言风格后生成新的文档,感兴趣的小伙伴可以了解一下... 目录核心思路技术路径实现步骤阶段一:准备工作阶段二:内容提取 (python 脚本)阶段三:语言风格调

go 指针接收者和值接收者的区别小结

《go指针接收者和值接收者的区别小结》在Go语言中,值接收者和指针接收者是方法定义中的两种接收者类型,本文主要介绍了go指针接收者和值接收者的区别小结,文中通过示例代码介绍的非常详细,需要的朋友们下... 目录go 指针接收者和值接收者的区别易错点辨析go 指针接收者和值接收者的区别指针接收者和值接收者的

Java Optional避免空指针异常的实现

《JavaOptional避免空指针异常的实现》空指针异常一直是困扰开发者的常见问题之一,本文主要介绍了JavaOptional避免空指针异常的实现,帮助开发者编写更健壮、可读性更高的代码,减少因... 目录一、Optional 概述二、Optional 的创建三、Optional 的常用方法四、Optio

C++中std::distance使用方法示例

《C++中std::distance使用方法示例》std::distance是C++标准库中的一个函数,用于计算两个迭代器之间的距离,本文主要介绍了C++中std::distance使用方法示例,具... 目录语法使用方式解释示例输出:其他说明:总结std::distance&n编程bsp;是 C++ 标准

使用Python实现表格字段智能去重

《使用Python实现表格字段智能去重》在数据分析和处理过程中,数据清洗是一个至关重要的步骤,其中字段去重是一个常见且关键的任务,下面我们看看如何使用Python进行表格字段智能去重吧... 目录一、引言二、数据重复问题的常见场景与影响三、python在数据清洗中的优势四、基于Python的表格字段智能去重