c++临时对象的探讨及相关性能提升

news2024/9/27 17:31:11

产生临时对象的情况

我们定义一个类进行测试


class tempVal {
public:
	int v1, v2;
	tempVal(int v1 = 0, int v2 = 0);
	tempVal(const tempVal& t) :v1(t.v1), v2(t.v2) 
	{
		cout << "调用拷贝构造函数" << endl;
		
	}
	virtual ~tempVal() 
	{
		cout << "调用析构函数" << endl;
	}
};

tempVal::tempVal(int v1, int v2) :v1(v1), v2(v2) 
{
	cout << "调用了构造函数" << endl;
	cout << "v1:" << v1 << endl;
	cout << "v2:" << v2 << endl;
}

以值传递的方式给函数传递参数

在文件中添加以下函数,其中函数的参数是我们定义的tempVal类,且使用的是值传递方式传参

int add(tempVal t)
{
	int tmp = t.v1 + t.v2;
	t.v1 = 1000;
	return tmp;
}

在main函数中添加代码

int main()
{
	tempVal tm(10, 20);
	int sum = add(tm);
	cout << "sum=" << sum << endl;
	cout << tm.v1 << endl;
    system("pause");
	return 0;
}

运行结果如下:

 观察上述输出结果,可以看到代码中输出了拷贝构造函数,这是因为调用Add成员函数时把对象tm传递给了Add函数,此时,系统会调用拷贝构造函数创建一个副本t(成员函数Add的形参),把tm对象复制给形参t,因为这是一个副本(复制),所以可以注意到,修改副本的val1的值为1000,并不会影响到外界tm对象的val1值(tm对象的val1值仍旧为10)

代码行中的形参t是一个局部对象(局部变量),从程序功能的角度来讲,函数体内需要临时使用它一下,来完成求和运算,严格意义上来讲,它不能称为一个临时对象,因为真正的临时对象往往指的是真实存在,但又感觉不到的对象(至少从代码上是不能直接看到的对象)。

但是代码生成了t对象,调用了tempVal类的拷贝构造函数,有了复制的动作,就会影响程序执行效率。修改代码以提升性能的方式也很简单,把传参方式修改为引用传参即可,即:

int add(tempVal& t)

再次运行观察结果:

观察上面的结果可以发现,少了一次调用拷贝构造函数和析构函数,提升了效率,如果对象很大,并且还从其他父类继承(继承会导致父类的拷贝构造函数也执行),那效率也许会提升很大,但是tm.value1的值在函数内部修改,直接被带到了函数外部,影响了函数外部tm对象的值,这就是引用的能力 

类型转换生成的临时对象

现在修改main函数内的代码:

	tempVal t1;
	t1=1000;

运行观察结果

观察上述结果发现,系统调用了两次构造函数,其中第一次是声明对象t1时调用的构造函数,而第二次构造函数的调用则是因为类型转换生成临时对象造成的

具体而言,系统会在此时生成一个临时对象(我们无法知道这个临时对象的名字和地址),之后调用构造函数把1000赋给这个临时对象的v1,而v2使用默认参数进行初始化再把这个临时对象赋值给t1,之后再调用析构函数是否掉生成的临时对象 (注意析构函数的调用销毁的是生成的临时对象,而不是t1,因为代码system("pause")的作用,此时对象t1还没有离开main函数的作用域)

为了观察方便,我们在tempVal的类中添加一个拷贝赋值运算符,如下:

tempVal& tempVal::operator=(const tempVal& tmp)
{
	cout<<"调用了拷贝赋值运算符"<<endl;
	this->v1=tmp.v1;
	this->v2=tmp.v2;
	return *this;
}

再次运行代码,观察运行结果:

现在,总结一下以上代码的运行过程:

  • 调用构造函数构造对象t1
  • 调用构造函数,将1000作为参数传递给构造函数,生成一个临时对象
  • 调用拷贝赋值运算符,将生成的临时对象赋值给对象t1
  • 是否掉临时对象

上述代码的优化也很简单,只需要将以上代码合并一下即可,如下:

tempVal t1=1000;

 此时,代码便少调用了一次构造函数、一次拷贝赋值运算符和一次析构函数

注意,针对“tempVal=1000;这行代码,这里的“=”不是赋值运算符,而是“定义时初始化”的概念。可以这样理解:在这里定义了t1对象,系统就为t1对象创建了预留空间,然后用1000调用构造函数来构造临时对象的时候,这种构造是在为t1对象创建的预留空间里进行的,所以并没有真的产生临时对象。

如果不想要隐式类型转换,将构造函数声明为explicit即可

再次观察一个由于隐式类型转换而造成生成不必要的临时对象的例子

我们在文件中定义一个新的函数,注意函数中参数的类型

void calc(const string& str)
{
	const char* p=str.c_str();
	return;
}

接下来,我们在main函数中调用这个函数,注意传给calc函数参数的类型

	char mystr[100]="I love China";
	calc(mystr);

运行代码,我们可以看到代码会正常运行成功。

但是我们看到,calc函数的形参类型为string,而我们传进去的参数类型为char数组,显然编译器为我们做了隐式类型转换,解决了代码运行过程中类型不匹配的问题,那么编译器是如何做的呢?

事实上,编译器首先调用string的构造函数生成了一个string类型的临时对象(通过我们的char数组实参mtstr进行初始化)之后形参str通过引用绑定到这个临时对象上,等函数调用结束之后,这个临时对象就被销毁了

显然,尽管我们在函数中对形参使用了左值引用,但是由于类型隐式转换的原因,我们仍旧在调用函数的过程中生成了不必要的临时对象,浪费了程序性能。

另外需要注意的是,假如我们将上述calc函数的形参中的const去掉,再次运行程序时就会报错。

造成代码报错的原因是,编译器在类型转换的过程中生成了临时对象,而形参str绑定在这个临时对象上,但是我们现在没有对形参str加const限制, 因此编译器就会认为我们有可能对str做出修改,而str绑定在临时对象上,就等于编译器认为我们可能对一个临时对象做修改,这是不被允许的。

因此,C++只会为const引用,而不会为非const引用产生临时对象

函数返回对象生成的临时对象

继续以本文最开始定义的tempVal类来说明问题

重新定义一个函数如下:

tempVal cal(tempVal& tmp)
{
	tempVal t;
	t.v1=tmp.v1*2;
	t.v2=tmp.v2*2;
	return t;
}

在main函数中调用该函数

int main()
{
	tempVal t1(10,20);
	cal(t1);
    system("pause");
	return 0;
}

 使用g++重新编译代码,并启用关闭编译优化选项

g++ main.cpp -o main.exe -fno-elide-constructors

运行可执行文件,观察终端输出结果

发现终端多输出了一次拷贝构造函数和两次析构函数的调用这是因为

  • cal函数调用结束时返回局部变量t的时候,会生成一个临时对象,把t的值拷贝给这个临时对象
  • 这个临时对象就可以在main函数中获取,从而获取到返回的值
  • 第一次析构函数的调用是cal函数结束时释放掉函数中的临时变量t
  • 第二次的析构函数的调用是main函数中(代码行55)调用cal函数结束后,返回的临时对象被释放掉了

注意,现代编译器会自动对返回值做优化,因此需要关闭优化选项才能看到返回的临时对象的生成

重新修改main函数的代码

int main()
{
	tempVal t1(10,20);
	tempVal t3=cal(t1);
    system("pause");
	return 0;
}

 再次运行观察结果

与上次相比,这次调用了两次拷贝构造函数,分析可知

  • 第一次是cal返回时产生临时对象调用的,作用是将cal函数中计算得到的变量t的值拷贝给这个临时对象返回出去
  • 第二次拷贝构造函数的调用则是将这个临时对象拷贝给t3从而构造对象t3时调用的 

右值引用在临时对象中的作用

我们继续修改代码以说明问题,将main函数的代码修改如下,也就是将t3修改为右值引用

int main()
{
	tempVal t1(10,20);
	tempVal&& t3=cal(t1);
    system("pause");
	return 0;
}

 观察运行结果发现,这次函数又只进行了一个拷贝构造函数的调用,并且少调用了一次析构函数

  • 显然该拷贝构造函数是由于返回值的临时对象产生的,但是在将这个临时对象赋值给t3的时候却没有调用拷贝构造函数
  • 这是因为使用了右值引用,将t3绑定到了cal函数返回的临时对象身上,从而避免了一次拷贝构造函数的调用
  • 并且从t3开始绑定这个返回的临时对象开始,它的生存周期将与t3的生存周期一样,这就是为什么与上次实验相比少了一次析构函数调用的原因

至此,我们已经通过代码可以“看到”右值引用的作用了

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

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

相关文章

14、强化学习Soft Actor-Critic算法:推导、理解与实战

基于LunarLander登陆器的Soft Actor-Critic强化学习&#xff08;含PYTHON工程&#xff09; Soft Actor-Critic算法是截至目前的T0级别的算法了&#xff0c;当前正在学习&#xff0c;在此记录一下下。 其他算法&#xff1a; 07、基于LunarLander登陆器的DQN强化学习案例&#…

第二百六十三回 给geolocator插件提交问题

文章目录 1. 知识回顾2. 问题描述与解决2.1 问题描述2.2 问题解决 3. 心得与感受 1. 知识回顾 我们在前面章回中介绍过如何获取位置信息&#xff0c;主要介绍的是geolocator这个三方包&#xff0c;不过在最近使用时却发现了问题&#xff0c;尝试搜索解决&#xff0c;但是没有结…

影像组学介绍

影像组学介绍 1 影像组学介绍2 具体提取影像组学方法流程及工具代码&#xff1a;2.1 影像数据获取2.2 感兴趣区域分割2.3 特征提取与降维选择2.3.1 特征提取&#xff1a;2.3.2 特征降维(特征选择) 2.4 建模分析&#xff1a;2.5 结果分析 参考&#xff1a; 1 影像组学介绍 其实…

7个向量数据库对比:Milvus、Pinecone、Vespa、Weaviate、Vald、GSI 和 Qdrant

本文简要总结了当今市场上正在积极开发的7个向量数据库&#xff0c;Milvus、Pinecone、Vespa、Weaviate、Vald、GSI 和 Qdrant 的详细比较。 我们已经接近在搜索引擎体验的基础层面上涉及机器学习&#xff1a;在多维多模态空间中编码对象。这与传统的关键字查找不同&#xff08…

探索设计模式的魅力:简单工厂模式

简单工厂模式&#xff08;Simple Factory Pattern&#xff09;是一种创建型设计模式&#xff0c;其主要目的是用于创建对象的实例。这种模式通过封装创建对象的代码来降低客户代码与具体类之间的耦合度。简单工厂不是GoF&#xff08;四人帮&#xff09;设计模式之一&#xff0c…

springBoot-自动配置原理

以下笔记内容&#xff0c; 整理自B站黑马springBoot视频&#xff0c;抖音Holis 1、自动配置原理 1.收集Spring开发者的编程习惯&#xff0c;整理开发过程使用的常用技术列表一>(技术集A) 2.收集常用技术(技术集A)的使用参数&#xff0c;整理开发过程中每个技术的常用设置列表…

NULL是什么?

NULL是一个编程术语&#xff0c;通常用于表示一个空值或无效值。在很多编程语言中&#xff0c;NULL用于表示一个变量或指针不引用任何有效的对象或内存位置。 NULL可以看作是一个特殊的值&#xff0c;表示缺少有效的数据或引用。当一个变量被赋予NULL值时&#xff0c;它表示该变…

Day28 17电话号码的字母组合 39组合求和 40组合求和II

17 电话号码的字母组合 给定一个仅包含数字 2-9 的字符串&#xff0c;返回所有它能表示的字母组合。 给出数字到字母的映射如下&#xff08;与电话按键相同&#xff09;。注意 1 不对应任何字母。 因为输入的数字的数量是不确定的&#xff0c;所以for循环的次数也是不确定的&…

基于JAVA的用户画像活动推荐系统 开源项目

目录 一、摘要1.1 项目介绍1.2 项目录屏 二、功能模块2.1 数据中心模块2.2 兴趣标签模块2.3 活动档案模块2.4 活动报名模块2.5 活动留言模块 三、系统设计3.1 用例设计3.2 业务流程设计3.3 数据流程设计3.4 E-R图设计 四、系统展示五、核心代码5.1 查询兴趣标签5.2 查询活动推荐…

Linux的权限(1)

目录 操作系统的"外壳"程序 外壳程序是什么&#xff1f; 为什么存在外壳程序&#xff1f; 外壳程序怎么运行操作&#xff1f; 权限 什么是权限&#xff1f; 权限的本质&#xff1f; Linux中的&#xff08;人&#xff09;用户权限&#xff1f; su和su -的区别…

安装rlwrap库出现问题

背景&#xff1a;oracle的sqlplus还是那么难用&#xff0c;不知道为什么不打包解决这个问题&#xff0c;留给用户&#xff0c;内核硬&#xff0c;就是猖狂。废话不多说。下载解压rlwrap-0.46.1.tar.gz;进入/tmp/database/rlwrap-0.46.1源码包&#xff0c;./configure checki…

大数据技术原理与应用期末复习(林子雨)

大数据技术原理与应用期末复习&#xff08;林子雨&#xff09; Hadoop的特性HBase编程实践NoSQL的四大类型键值数据库优点&#xff1a;缺点&#xff1a; 列族数据库优点&#xff1a;缺点&#xff1a; 文档数据库优点&#xff1a;缺点&#xff1a; 图数据库优点&#xff1a;缺点…

计算机网络面试八股复习:常见的Http状态码

前言 面试被问到过一次。自己最近使用Gin框架&#xff0c;在Response的时候有时候也会用到一个自定义的状态码。因此归纳一下这方面&#xff0c;供自己日后面试复习以及开发时候参考。 HTTP 全名“超文本传输协议”&#xff08;我也不懂为什么面试官问这个…&#xff09; 属…

TypeScript基础(二)扩展类型-枚举及其位运算

✨ 专栏介绍 TypeScript是一种由微软开发的开源编程语言&#xff0c;它是JavaScript的超集&#xff0c;意味着任何有效的JavaScript代码都是有效的TypeScript代码。TypeScript通过添加静态类型和其他特性来增强JavaScript&#xff0c;使其更适合大型项目和团队开发。 在TypeS…

springboot实现黑名单和白名单功能

题外话 关于黑名单和白名单功能&#xff0c;我觉得可以直接用linux服务器的iptables或nftables来实现黑名单和白名单功能。这两个工具都是Linux系统上用于配置防火墙规则的命令行工具。 iptables&#xff1a; 描述&#xff1a; iptables 是一个用于配置IPv4数据包过滤规则的工具…

Spring Task 任务调度工具

大家好我是苏麟 , 今天聊聊Spring Task 任务调度工具 Spring Task Spring Task 是Spring框架提供的任务调度工具&#xff0c;可以按照约定的时间自动执行某个代码逻辑。 定位&#xff1a;定时任务框架 作用&#xff1a;定时自动执行某段Java代码 什么是定时任务 ? 通过时…

uni-app做A-Z排序通讯录、索引列表

上图是效果图&#xff0c;三个问题 访问电话通讯录&#xff0c;拿数据拿到用户的联系人数组对象&#xff0c;之后根据A-Z排序根据字母索引快速搜索 首先说数据怎么拿 - 社区有指导https://ask.dcloud.net.cn/question/64117 uniapp 调取通讯录 // #ifdef APP-PLUSplus.contac…

怎么把身份证压缩到200k以下?一分钟教你如图片压缩

在网络平台办理一些业务的时候&#xff0c;经常会需要上传我们的身份证照片&#xff0c;但是大多数平台为了用户体验&#xff0c;会限制上传的图片大小&#xff0c;比如图片不得超过200kb&#xff0c;当我们提交的身份证图片超出限制&#xff0c;就无法顺利提交&#xff1b;这时…

每日一练:LeeCode-104. 二叉树的最大深度【二叉树】

本文是力扣LeeCode-104. 二叉树的最大深度 学习与理解过程&#xff0c;本文仅做学习之用&#xff0c;对本题感兴趣的小伙伴可以出门左拐LeeCode。 给定一个二叉树 root &#xff0c;返回其最大深度。 二叉树的 最大深度 是指从根节点到最远叶子节点的最长路径上的节点数。 示例…

Asp .Net Web应用程序(.Net Framework4.8)网站发布到IIS

开启IIS 如果已开启跳过这步 打开控制面板-程序 打开IIS 发布Web程序&#xff08;.Net Framework 4.8 web网页&#xff09; 进入IIS管理器新建一个应用池 新建一个网站 网站创建完毕 为文件夹添加访问权限 如果不添加访问权限&#xff0c;运行时将会得到如下错误 设置权限 勾…