C++Primer学习(13.2 拷贝控制和资源管理)

13.2 拷贝控制和资源管理
通常,管理类外资源的类必须定义拷贝控制成员。如我们在13.1.4节(第447页)中所见,这种类需要通过析构函数来释放对象所分配的资源。一旦一个类需要析构函数,那么它几乎肯定也需要一个拷贝构造函数和一个拷贝赋值运算符。
为了定义这些成员,我们首先必须确定此类型对象的拷贝语义。一般来说,有两种选择:可以定义拷贝操作,使类的行为看起来像一个值或者像一个指针。
类的行为像一个值,意味着它应该也有自己的状态。当我们拷贝一个像值的对象时,副本和原对象是完全独立的。改变副本不会对原对象有任何影响,反之亦然。
行为像指针的类则共享状态。当我们拷贝一个这种类的对象时,副本和原对象使用相同的底层数据。改变副本也会改变原对象,反之亦然。
在我们使用过的标准库类中,标准库容器和string 类的行为像一个值。而不出意外的,shared_ptr类提供类似指针的行为,就像我们的strBlob类(参见12.11节,第405 页)一样,I0类型和 unique_ptr不允许拷贝或赋值,因此它们的行为既不像值也不像指针。
为了说明这两种方式,我们会为练习中的HasPtr类定义拷贝控制成员。首先,我们将令类的行为像一个值;然后重新实现类,使它的行为像一个指针。
我们的 HasPtr类有两个成员,一个int和一个 string 指针。通常,类直接拷贝内置类型(不包括指针)成员:这些成员本身就是值,因此通常应该让它们的行为像值样。我们如何拷贝指针成员决定了像Hasptr这样的类是具有类值行为还是类指针行为。
13.2.1 行为像值的类
为了提供类值的行为,对于类管理的资源,每个对象都应该拥有一份自己的拷贝。这意味着对于ps指向的string,每个HasPtr对象都必须有自己的拷贝。为了实现类值行为,HasPtr 需要
(1)定义一个拷贝构造函数,完成string的拷贝,而不是拷贝指针
(2)定义一个析构函数来释放string
(3)定义一个拷贝赋值运算符来释放对象当前的string,并从右侧运算对象拷贝string
类值版本的 HasPtr如下所示

class HasPtr
{
public:HasPtr(const std::string &s=std::string()):ps(new std::string(s))i(0){}//构造函数//对ps指向的string,每个HasPtr对象都有自己的拷贝HasPtr(const HasPtr &p):ps(new std::string(*p.ps))i(p.i){}//拷贝构造函数HasPtr & operator=(const HasPtr&);~HasPtr(){delete ps;}//析构函数
private:std::string *pS;int i;
}

我们的类足够简单,在类内就已定义了除赋值运算符之外的所有成员函数。第一个构造函数接受一个(可选的)string参数。这个构造函数动态分配它自己的string副本,并将指向string的指针保存在ps中。拷贝构造函数也分配它自己的string副本。析构函数对指针成员ps执行delete,释放构造函数中分配的内存。
类值拷贝赋值运算符
赋值运算符通常组合了析构函数和构造函数的操作。类似析构函数,赋值操作会销毁左侧运算对象的资源。类似拷贝构造函数,赋值操作会从右侧运算对象拷贝数据。但是,非常重要的一点是,这些操作是以正确的顺序执行的,即使将一个对象赋予它自身,也保证正确。而且,如果可能,我们编写的赋值运算符还应该是异常安全的–当异常发生时能将左侧运算对象置于一个有意义的状态。
在本例中,通过先拷贝右侧运算对象,我们可以处理自赋值情况,并能保证在异常发生时代码也是安全的。在完成拷贝后,我们释放左侧运算对象的资源,并更新指针指向新分配的 string:

HasPtr& HasPtr::operator=(const HasPtr &rhs)
{auto newp = new string(*rhs.ps);//拷贝底层stringps = newp;//释放旧内存delete ps;i= rhs.i;//从右侧运算对象拷贝数据到本对象return *this;//返回本对象
}

在这个赋值运算符中,非常清楚,我们首先进行了构造函数的工作: newp的初始化器等价于 HasPtr的拷贝构造函数中ps的初始化器。接下来与析构函数一样,我们 delete当前 ps 指向的 string。然后就只剩下拷贝指向新分配的string的指针,以及从rhs拷贝int值到本对象了。
关键概念:赋值运算符
当你编写赋值运算符时,有两点需要记住:
(1)如果将一个对象赋予它自身,赋值运算符必须能正确工作。
(2)大多数赋值运算符组合了析构函数和拷贝构造函数的工作。
当你编写一个赋值运算符时,一个好的模式是先将右侧运算对象拷贝到一个局部临时对象中。当拷贝完成后,销毁左侧运算对象的现有成员就是安全的了。一旦左侧运算对象的资源被销毁,就只剩下将数据从临时对象拷贝到左侧运算对象的成员中了。
为了说明防范自赋值操作的重要性,考虑如果赋值运算符如下编写将会发生什么//这样编写赋值运算符是错误的!

HasPtr& HasPtr::operator=(const HasPtr &rhs)
{delete ps;//释放对象指向的string//如果rhs和*this 是同一个对象,我们就将从已释放的内存中拷贝数据!ps = new string(*(rhs.ps));i = rhs.i;return *this;
}

如果 rhs 和本对象是同一个对象,delete ps 会释放this和rhs指向的string。接下来,当我们在new表达式中试图拷贝(rhs.ps)时,就会访问一个指向无效内存的指针,其行为和结果是未定义的。
NARNING:
对于一个赋值运算符来说,正确工作是非常重要的,即使是将一个对象赋予它自身,也要能正确工作。一个好的方法是在销毁左侧运算对象资源之前拷贝右侧运算对象。
13.2.2 定义行为像指针的类
对于行为类似指针的类,我们需要为其定义拷贝构造函数和拷贝赋值运算符,来拷贝指针成员本身而不是它指向的string。我们的类仍然需要自己的析构函数来释放接受string 参数的构造函数分配的内存。但是,在本例中,构函数不能单方面地释放关联的string。只有当最后一个指向string的HasPtr 销毁时,它才可以释放string。
令一个类展现类似指针的行为的最好方法是使用shared_ptr来管理类中的资源拷贝(或赋值)一个shared_ptr会拷贝(赋值)shared_ptr所指向的指针。sharedptr类自己记录有多少用户共享它所指向的对象。当没有用户使用对象时shared_ptr类负责释放资源。
但是,有时我们希望直接管理资源。在这种情况下,使用引用计数(reference count)(参见12.1.1节,第402页)就很有用了。为了说明引用计数如何工作,我们将重新定义HasPtr,令其行为像指针一样,但我们不使用shared_ptr,而是设计自己的引用计数。
引用计数
引用计数的工作方式如下:
(1)除了初始化对象外,每个构造函数(拷贝构造函数除外)还要创建一个引用计数用来记录有多少对象与正在创建的对象共享状态。当我们创建一个对象时,只有个对象共享状态,因此将计数器初始化为1。
(2)拷贝构造函数不分配新的计数器,而是拷贝给定对象的数据成员,包括计数器。拷贝构造函数递增共享的计数器,指出给定对象的状态又被一个新用户所共享。
(3)析构函数递减计数器,指出共享状态的用户少了一个。如果计数器变为0,则析构函数释放状态。
(4)拷贝赋值运算符递增右侧运算对象的计数器,递减左侧运算对象的计数器。如果左侧运算对象的计数器变为0,意味着它的共享状态没有用户了,拷贝赋值运算符就必须销毁状态。
唯一的难题是确定在哪里存放引用计数。计数器不能直接作为HasPtr对象的成员下面的例子说明了原因:

HasPtr p1("Hiya!");
HasPtr p2(p1);//p1和p2指向相同的string
HasPtr p3(p1);// p1、p2和p3 都指向相同的 string

如果引用计数保存在每个对象中,当创建p3时我们应该如何正确更新它呢?可以递增p1中的计数器并将其拷贝到p3中,但如何更新p2中的计数器呢?
解决此问题的一种方法是将计数器保存在动态内存中。当创建一个对象时,我们也分配一个新的计数器。当拷贝或赋值对象时,我们拷贝指向计数器的指针。使用这种方法,副本和原对象都会指向相同的计数器。
定义一个使用引用计数的类
通过使用引用计数,我们就可以编写类指针的Hasptr版本了:

class HasPtr
{
public ://构造函数分配新的string和新的计数器,将计数器置为1HasPtr(const std::string &s=std::string()): ps(new std::string(s))i(0),use(new std::size t(1)){}//拷贝构造函数拷贝所有三个数据成员,并递增计数器HasPtr(const HasPtr &p):ps(p.ps)i(p.i)use(p.use){++*use;}HasPtr& operator=(const HasPtr&);~HasPtr();
private:std::string *ps;int i;std::size_t*use;//用来记录有多少个对象共享*ps的成员
}

在此,我们添加了一个名为use的数据成员,它记录有多少对象共享相同的string。接受 string 参数的构造函数分配新的计数器,并将其初始化为1,指出当前有一个用户使用本对象的 string 成员。
类指针的拷贝成员“篡改”引用计数
当拷贝或赋值一个HasPtr对象时,我们希望副本和原对象都指向相同的string。即,当拷贝一个Hasptr时,我们将拷贝ps本身,而不是ps指向的string。当我们进行拷贝时,还会递增该string关联的计数器(我们在类内定义的)拷贝构造函数拷贝给定HasPtr的所有三个数据成员。这个构造函数还递增use 成员,指出ps 和p.ps 指向的string 又有了一个新的用户。
析构函数不能无条件地 deleteps-可能还有其他对象指向这块内存。析构函数应该递减引用计数,指出共享string的对象少了一个。如果计数器变为0,则析构函数释放ps 和use指向的内存:

HasPtr::~HasPtr()
{if(--*use == 0)//如果引用计数变为0{delete ps;//释放 string 内存delete use;//释放计数器内存}
}

拷贝赋值运算符与往常一样执行类似拷贝构造函数和析构函数的工作。即,它必须递增右侧运算对象的引用计数(即,拷贝构造函数的工作),并递减左侧运算对象的引用计数,在必要时释放使用的内存(即,析构函数的工作)。
而且与往常一样,赋值运算符必须处理自赋值。我们通过先递增rhs中的计数然后再递减左侧运算对象中的计数来实现这一点。通过这种方法,当两个对象相同时,在我们检查ps(及use)是否应该释放之前,计数器就已经被递增过了:

HasPtr& HasPtr::operator=(const HasPtr &rhs)
{++*rhs.use;//递增右侧运算对象的引用计数if(--*use==0)//然后递减本对象的引用计数{delete ps;//如果没有其他用户delete use;//释放本对象分配的成员}ps = rhs.ps;//将数据从rhs拷贝到本对象i = rhs.i;use =rhs.use;return *this;//返回本对象
}

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

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

相关文章

Bulk Rename Utility(BRU)——大批量重命名实用程序

Bulk Rename Utility(BRU)——大批量重命名实用程序 博主要给博客网站搞博客封面,几百张图没编号,一弄这个就好了,亲测十分好用,下面的b站教程更是一绝,快快使用起来 文章目录 Bulk Rename Ut…

鸿蒙生态开发

鸿蒙生态开发概述 鸿蒙生态是华为基于开源鸿蒙(OpenHarmony)构建的分布式操作系统生态,旨在通过开放共享的模式连接智能终端设备、操作系统和应用服务,覆盖消费电子、工业物联网、智能家居等多个领域。以下从定义与架构、核心技术…

Matlab概率区间预测全家桶更新了,新增光伏出力区间预测,4种分布可供预测

基本介绍 适用于matlab2020及以上。可任意选择置信区间,区间覆盖率picp、区间平均宽度百分比等等,可用于预测不确定性,效果如图所示,采用KDE,4种分布进行预测,有对比,可以替换成自己的数据。 …

C语言【文件操作】详解中(会使用fgetc,fputc,fgets,fputs,fscanf,fprintf,fread,fwrite函数)

引言 介绍和文件操作中文件的顺序读写相关的函数 看这篇博文前,希望您先仔细看一下这篇博文,理解一下文件指针和流的概念:C语言【文件操作】详解上-CSDN博客文章浏览阅读606次,点赞26次,收藏4次。先整体认识一下文件是…

深入剖析Java虚拟机(JVM):从零开始掌握Java核心引擎

📌 引言:为什么每个Java开发者都要懂JVM? 想象你是一名赛车手,Java是你的赛车,而JVM就是赛车的引擎。 虽然你可以不关心引擎内部构造就能开车,但要想在比赛中获胜,必须了解引擎如何工作&#…

鸿蒙harmonyOS笔记:练习CheckBoxGroup获取选中的值

除了视觉效果实现全选和反选以外,咱们经常需要获取选中的值,接下来看看如何实现。 核心步骤: 1. 给 CheckBoxGroup 注册 onChange。 2. CheckBox 添加 name 属性。 3. 在 onChange 的回调函数中获取 选中的 name 属性。 事件&#xff1a…

通俗易懂搞懂@RequestParam 和 @RequestBody

📌 博主简介: 💻 努力学习的 23 级科班生一枚 🚀🏠 博主主页 : 📎 灰阳阳📚 往期回顾 :Session和Cookie我不允许你不懂💬 每日一言: 「流水不争先&#xff0c…

Flink 内存管理

一、内存模型 上图是一个 Flink 程序进程总体的内存模型,其包含 Flink 使用内存、JVM 元空间以及 JVM 开销。 Flink 使用了堆上内存和堆外内存;框架内存使用了堆上内存和堆外内存的直接内存;Task 使用堆上内存和堆外内存的直接内存;管理内存、JVM 元空间以及 JVM 内存开销使…

【工具变量】中国各地级市是否属于“信息惠民国家试点城市”匹配数据(2010-2024年)

数据来源:国家等12部门联合发布的《关于加快实施信息惠民工程有关工作的通知》 数据说明:内含原始文件和匹配结果,当试点城市在2014年及以后,赋值为1;试点城市在2014年之前或该城市从未实施信息惠民试点工程&#x…

git的底层原理

git的底层原理 三段话总结git, 1. 工作原理:git管理是一个DAG有向无环图,HEAD指针指向branch或直接指向commit,branch指向commit,commit指向tree,tree指向别的tree或直接指向blob。 2. git所管理的一个目录…

安装React开发者工具

我们在说组件之前,需要先安装一下React官方推出的开发者工具,首先我们分享在线安装方式 首先打开谷歌网上应用商店(针对谷歌浏览器),在输入框内搜索react,安装如下插件: 注意安装提供方为Facebook的插件,这…

排列与二进制

#include<iostream> using namespace std; int count_two(int n,int m){int count0;for(int i0;i<m;i){ //统计2的因子个数 int numn-i;while(num%20){count;num /2;}}return count; } int main(){int n,m;while(1){cin >> n >> m;if(n0 && m0)br…

鱼书--学习2

6. 与学习相关的技巧 6.1 参数的更新 &#xff08;1&#xff09; SGD的缺点&#xff1a;SGD低效的根本原因是&#xff0c;梯度的方向并没有指向最小值的方向 基于SGD的最优化的更新路径&#xff1a;呈“之”字形朝最小值(0, 0)移动&#xff0c;效率低 &#xff08;2&#x…

基于SSM框架的汽车租赁平台(源码+lw+部署文档+讲解),源码可白嫖!

摘要 时代在飞速进步&#xff0c;每个行业都在努力发展现在先进技术&#xff0c;通过这些先进的技术来提高自己的水平和优势&#xff0c;汽车租赁平台当然不能排除在外。汽车租赁平台是在实际应用和软件工程的开发原理之上&#xff0c;运用Java语言以及SSM框架进行开发&#x…

LangChain Chat Model学习笔记

Prompt templates: Few shot、Example selector 一、Few shot(少量示例) 创建少量示例的格式化程序 创建一个简单的提示模板&#xff0c;用于在生成时向模型提供示例输入和输出。向LLM提供少量这样的示例被称为少量示例&#xff0c;这是一种简单但强大的指导生成的方式&…

新配置了一台服务器+域名共178:整个安装步骤,恢复服务

买了一台服务器域名eesou.com&#xff1a; 服务器选的是99元最低配的&#xff0c;用免费的镜像&#xff1a;宝塔面板 eesou.com是一口价买的 79&#xff0c;原来wjsou.com卖了。 原来的配置全丢了。开始重新安装步骤。 域名备案才能用&#xff0c;提交就等着了 服务器配置 …

Netty——BIO、NIO 与 Netty

文章目录 1. 介绍1.1 BIO1.1.1 概念1.1.2 工作原理1.1.3 优缺点 1.2 NIO1.2.1 概念1.2.2 工作原理1.2.3 优缺点 1.3 Netty1.3.1 概念1.3.2 工作原理1.3.3 优点 2. Netty 与 Java NIO 的区别2.1 抽象层次2.2 API 易用性2.3 性能优化2.4 功能扩展性2.5 线程模型2.6 适用场景 3. 总…

我的uniapp自定义模板

uniapp自定义模板 如有纰漏请谅解&#xff0c;以官方文档为准后面这段时间我会学习小程序开发的知识&#xff0c;会持续更新可以查看我的github&#xff0c;后续我会上传我的uniapp相关练习代码有兴趣的话可以浏览我的个人网站&#xff0c;我会在上面持续更新内容&#xff0c;…

Wispr Flow,AI语言转文字工具

Wispr Flow是什么 Wispr Flow 是AI语音转文本工具&#xff0c;基于先进的AI技术&#xff0c;帮助用户在任何应用程序中实现快速语音转文字。 Wispr Flow支持100多种语言&#xff0c;具备自动编辑、上下文感知和低音量识别等功能&#xff0c;大幅提升写作和沟通效率。Wispr Fl…

美国国家数据浮标中心(NDBC)

No.大剑师精品GIS教程推荐0地图渲染基础- 【WebGL 教程】 - 【Canvas 教程】 - 【SVG 教程】 1Openlayers 【入门教程】 - 【源代码示例 300】 2Leaflet 【入门教程】 - 【源代码图文示例 150】 3MapboxGL【入门教程】 - 【源代码图文示例150】 4Cesium 【入门教程】…