C++相关概念和易错语法(8)(匿名对象、构造+拷贝构造优化、构造析构顺序)

news2024/11/16 5:40:09

1.匿名对象

当我们实例化对象后,有的对象可能只使用一次,之后就没用了。这个时候我们往往要主动去析构它,否则会占着浪费空间。但是如果遇到大量的这种情况,我们并不想每次都去创建对象、调用、析构,这样会写出很多重复无聊的代码,为了精简,可以考虑使用匿名对象

下面是它使用的一个实例:


#include <iostream>
using namespace std;

class A
{
public:
	A(int a = 0, int b = 0)
		:_a(a)
		,_b(b)
	{
		cout << "A(int a = 0, int b = 0)" << endl;
	}

	~A()
	{
		cout << "~A()" << endl;
	}

	void Fun()
	{
		cout << "调用函数" << endl;
	}
private:
	int _a;
	int _b;
};

int main()
{
	cout << "在这之后构造\n" << endl;

	A().Fun();

	cout << "\n在这之前析构" << endl;

	return 0;
}
wAAACH5BAEKAAAALAAAAAABAAEAAAICRAEAOw==

运行的结果是

我们可以发现,匿名对象的生命周期就在那一行,在调用之后就会立马销毁。这会让我们在某些情况下代码量大大减少,看上去也很直观。

注意书写格式:

2.构造函数、拷贝构造、析构函数的优化

对于任何表达式,它都会有一个返回值,这个返回值是一个临时常变量,如果有变量接收就赋值给它,如果没有就抛弃。对于类和对象的相关操作理论上也应如此。但和内置类型不同,对象每次进行一次复制需要的开销都很大,对性能影响很大,所以编译器会做出优化,简化现有逻辑,提高效率。

下面是一些常见的在VS2022上Debug下的优化:

对于优化问题,我们常常讨论的是构造次数,因为析构和构造成对出现。

主要的优化类型是:构造+拷贝构造->构造

a.隐式类型转换


#include <iostream>
using namespace std;

class A
{

public:
	A(int a = 0)
		:_a(a)
	{
		cout << "A(int a = 0)" << endl;
	}

	A(const A& a)
		: _a(a._a)
	{
		cout << "A(A& a)" << endl;
	}

	~A()
	{
		cout << "~A()" << endl;
	}

private:
	int _a;
};

int main()
{
	A a = 1;

	return 0;
}

对于A a = 1,一般逻辑是先将1作为形参调用单参数的A的构造函数得到临时常对象tmp,再采用拷贝构造给a初始化。

但是优化之后就不会生成这个临时变量了。编译器会直接将1作为参数调用对象a对应的构造函数进行初始化。这样的话就少了一次拷贝。

b.隐式类型转换+调用函数


#include <iostream>
using namespace std;

class A
{

public:
	A(int a = 0)
		:_a(a)
	{
		cout << "A(int a = 0)" << endl;
	}

	A(const A& a)
		: _a(a._a)
	{
		cout << "A(A& a)" << endl;
	}

	~A()
	{
		cout << "~A()" << endl;
	}

	static	void Fun(A a)
	{
		cout << "Fun(A a)" << endl;
	}

private:
	int _a;
};

int main()
{
	A::Fun(1);

	return 0;
}

在这段代码中,1应该先调用构造生成临时对象,再将这个对象拷贝构造给形参a。

但是在这里,我们就直接将1作为参数去调用a的对应的构造函数,就不会去生成那个临时变量了。

c.当匿名对象作为实参


#include <iostream>
using namespace std;

class A
{

public:
	A(int a = 0)
		:_a(a)
	{
		cout << "A(int a = 0)" << endl;
	}

	A(const A& a)
		: _a(a._a)
	{
		cout << "A(A& a)" << endl;
	}

	~A()
	{
		cout << "~A()" << endl;
	}

	static	void Fun(A a)
	{
		cout << "Fun(A a)" << endl;
	}

private:
	int _a;
};

int main()
{
	A::Fun(A(1));

	return 0;
}

这里,本来的逻辑是1先去调用匿名对象的构造函数,完成匿名对象的初始化;然后匿名对象再去调用形参a的拷贝构造,完成a的初始化。

但是我们知道匿名对象是即用即扔的,它和临时对象没什么区别,都只在当前行有作用,因此这里依然是跳过构造匿名对象,直接将1作为参数去调用形参的构造函数。

借助这个优化,我们可以减少代码量


int main()
{
	A::Fun(A(1));//两种写法相同
	A::Fun(1);

	return 0;
}

一般来说采用后者的写法,毕竟要写的代码量更小,而且提高可读性。

d.函数返回值是自定义类型


#include <iostream>
using namespace std;

class A
{

public:
	A(int a = 0)
		:_a(a)
	{
		cout << "A(int a = 0)" << endl;
	}

	A(const A& a)
		: _a(a._a)
	{
		cout << "A(A& a)" << endl;
	}

	~A()
	{
		cout << "~A()" << endl;
	}

	static	A Fun()
	{
		cout << "Fun(A a)" << endl;
		return A(2);
	}

private:
	int _a;
};

int main()
{
	A ret = A::Fun();

	return 0;
}

在调用完函数后,函数应该返回值并赋值。这里的逻辑应该是先将2作为参数去调用匿名对象的构造函数,然后将这个匿名对象拷贝构造给ret,使其初始化。

在优化后,我们直接将2作为参数去调用ret的构造函数。

但是,有的情况并不会优化,主要是看构造函数和拷贝构造有没有在一行代码中连续调用。



#include <iostream>
using namespace std;

class A
{

public:
	A(int a = 0)
		:_a(a)
	{
		cout << "A(int a = 0)" << endl;
	}

	A(const A& a)
		: _a(a._a)
	{
		cout << "A(A& a)" << endl;
	}

	~A()
	{
		cout << "~A()" << endl;
	}

	A& operator= (const A& tmp)
	{
		cout << "A& operator= (const A& tmp)" << endl;
		_a = tmp._a;
		return *this;
	}

	static	A Fun()
	{
		cout << "Fun(A a)" << endl;
		return A(2);
	}

private:
	int _a;
};

int main()
{
	A ret;

	ret = A::Fun();

	return 0;
}

很明显,这里ret的初始化在调用函数前一行,会直接调用它的构造函数。之后进入函数,返回值需要再调用一次匿名对象的构造函数 ,之后调用的是赋值重载运算符。

在这里没有使用构造+拷贝构造,所以没有可优化的地方,一切执行逻辑和我们的推导逻辑一致。

3.构造和析构的顺序

(1)全局对象最先构造,在程序开始时就按顺序构造。

(2)局部static对象在第一次调用才构造,第二次就不构造了。

(3)普通的局部对象按顺序构造

(4)除了static修饰的局部对象以外,所有对象(局部、全局)先构造的后析构。

(5)static修饰的局部对象因为生命周期发生了改变,在所有局部对象都析构之后,在全局对象析构之前按“栈”的规则析构

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

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

相关文章

【Linux】对system V本地通信的内核级理解

一、system V版本的进程间通信技术 通过之前的学习&#xff0c;我们大致可以感受出来&#xff0c;共享内存&#xff0c;消息队列和信号量在使用的时候是有很多共性的。它们三个的接口&#xff0c;包括接口中传的参数有的都有很大的相似度。其实&#xff0c;共享内存&#xff…

大功率岸电电源技术研究及发展趋势

大功率岸电电源是一种利用海洋潮汐、波浪等可再生能源进行电能充电的设备&#xff0c;也称为海洋能充电器或潮汐能发电机。它通过接收潮汐、波浪等可再生能源&#xff0c;将可再生能源转换为电能进行充电&#xff0c;为港口、岛屿等地区提供清洁、高效的电力。大功率岸电电源对…

SpringSecurity源码4

SecurityContext.class 当前线程关联的最小安全信息&#xff0c;提供Authentication的get/set方法 SecurityContextHolder.class SecurityContext的持有器 // 全部委托给策略类public static void setContext(SecurityContext context) {strategy.setContext(context);}public …

算法课程笔记——蓝桥云课第二次直播

注意是‘’ 都正确 可以理解为a的首地址也是数字&#xff0c;向右1 %p逻辑地址 Cin cout字符串“”单个字符本身‘’&#xff0c;其他时候不用加 这样就可以 逆运算 bool比较真假<从小到大排 11/25 都输出最省事 变成长度为n1的数组 考虑到整个都可能为一个颜色&#xff0c;…

UML/SysML建模工具更新情况-截至2024年4月(1)5款-Trufun建模平台 v2024

DDD领域驱动设计批评文集 做强化自测题获得“软件方法建模师”称号 《软件方法》各章合集 工具最新版本&#xff1a;itemis CREATE 5.2.2 更新时间 2024年3月22日 工具简介 原名YAKINDU Statechart Tools。状态机建模工具&#xff0c;支持各种语言的代码生成&#xff0c;提…

Semaphore信号量源码解读与使用

&#x1f3f7;️个人主页&#xff1a;牵着猫散步的鼠鼠 &#x1f3f7;️系列专栏&#xff1a;Java全栈-专栏 &#x1f3f7;️个人学习笔记&#xff0c;若有缺误&#xff0c;欢迎评论区指正 目录 1. 前言 2. 什么是Semaphore&#xff1f; 3. Semaphore源码解读 3.1 acquire…

yml文件解析

.yml 后缀的文件可以有多个application.yml # 项目相关配置 用于 RuoYiConfig.java ruoyi:# 名称name: RuoYi# 版本version: 3.8.5# 版权年份copyrightYear: 2023# 实例演示开关demoEnabled: true# 文件路径 示例&#xff08; Windows配置D:/ruoyi/uploadPath&#xff0c;Lin…

SQLite的知名用户(二十九)

返回&#xff1a;SQLite—系列文章目录 上一篇:SQLite作为应用程序文件格式&#xff08;二十八&#xff09; 下一篇:SQLite FTS5 扩展&#xff08;三十&#xff09; SQLite被数以百万计的应用程序使用 从字面上看&#xff0c;有数十亿次部署。 SQLite 是 当今世界。 下面…

C语言指针加法/减法

减法运算 指针的减法运算公式&#xff1a;就是两个指针的内存差值 / 指向的数据类型的内存空间大小 如果改变了指针类型指向的数据类型&#xff0c;对指针进行 - 运算&#xff0c;其新指向的位置只和转换后的数据类型有关 如果是char &#xff0c; 则指针指向变化1byte 如果…

【微服务】spring读取配置文件多种方式深入详解

目录 一、前言 二、java配置文件介绍 2.1 java配置文件产生原因 2.2 项目使用配置文件好处 2.3 springboot项目配置文件的必要性 2.4 微服务架构下配置文件使用场景 三、java读取配置文件常用方法 3.1 使用Properties类读取配置文件 3.1.1 使用getResourceAsStream读取…

【人工智能基础】知识表示和专家系统

目录 一、知识 知识 产生式表示法 产生式表示形式 确定的事实性知识的产生式表示 不确定的事实性知识的产生式表示 确定的规则知识的产生式表示 不确定的规则知识的产生式表示 产生式系统构成 优点 缺点 语义网络 优点 缺点 框架 框架的一般结构 框架的继承 优…

如何30天快速掌握键盘盲打

失业后在家备考公务员&#xff0c;发现了自己不正确的打字方式&#xff0c;决定每天抽出一点时间练习打字。在抖音上看到一些高手的飞速盲打键盘后&#xff0c;觉得使用正确的指法打字是很必要的。 练习打字&#xff0c;掌握正确的键盘指法十分关键。 练习打字的第一步是找到…

AJAX——ajax原理

1.XMLHttpRequest 定义&#xff1a;XMLHttpRequest&#xff08;XHR&#xff09;对象用于与服务器交互。通过XMLHttpRequest可以在不刷新页面的情况下请求特定URL&#xff0c;获取数据。这允许网页在不影响用户操作的情况下&#xff0c;更新页面的局部内容。XMLHttpRequest在AJA…

大话设计模式-里氏代换原则

里氏代换原则&#xff08;Liskov Substitution Principle&#xff0c;LSP&#xff09; 概念 里氏代换原则是面向对象设计的基本原则之一&#xff0c;由美国计算机科学家芭芭拉利斯科夫&#xff08;Barbara Liskov&#xff09;提出。这个原则定义了子类型之间的关系&#xff0…

MySQL之binlog归档日志

binlog&#xff08;二进制归档日志&#xff09; binlog 二进制日志记录保存所有执行过的修改操作语句&#xff0c;不保存查询操作。如果 MySQL 服务意外停止&#xff0c;可通过二进制日志文件排查&#xff0c;用户操作或表结构操作&#xff0c;从而来恢复数据库数据。启动 bin…

[BJDCTF 2020]encode

脱壳 rc4 自定义base64 异或

Make/Makefile详解

文章目录 make/MakefilemakeMakefile时间 make/Makefile Linux项目自动化构建工具。makefile定义了一系列的规则来指定&#xff0c;哪些文件需要先编译&#xff0c;哪些文件需要后编译&#xff0c;哪些文件需要重新编译&#xff0c;甚至进行更复杂的功能。makefile带来的好处就…

神经网络中的神经元和激活函数介绍

文章目录 1、什么是人工神经网络 2、什么是神经元 3、什么是激活函数 线性激活函数 Sigmoid激活函数 双曲正切激活函数 修正线性单元&#xff08;ReLU&#xff09;激活函数 Leaky ReLU激活函数 Softmax激活函数 1、什么是人工神经网络 神经网络能够利用多层神经元学习复杂的模…

回顾发展史,WEB组态终于可以搭建业务系统了!

一、WEB组态融合了传统工业组态和现代数据大屏 组态软件在工业控制领域有着二三十年的历史&#xff0c;比较知名的国内有组态王、国外有MCGS/WinCC等&#xff0c;用来搭建上位机监控画面&#xff0c;通常配合PLC使用&#xff0c;通过配置的方式&#xff0c;不需要编码&#xf…

线性表的链式存储(双向循环链表)

文章目录 前言一、双向循环链表是什么&#xff1f;二、双向循环链表的意义三、双向循环链表的操作实现总结 前言 T_T此专栏用于记录数据结构及算法的&#xff08;痛苦&#xff09;学习历程&#xff0c;便于日后复习&#xff08;这种事情不要啊&#xff09;。所用教材为《数据结…