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

本文主要是介绍C++相关概念和易错语法(8)(匿名对象、构造+拷贝构造优化、构造析构顺序),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

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修饰的局部对象因为生命周期发生了改变,在所有局部对象都析构之后,在全局对象析构之前按“栈”的规则析构

这篇关于C++相关概念和易错语法(8)(匿名对象、构造+拷贝构造优化、构造析构顺序)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

C++右移运算符的一个小坑及解决

《C++右移运算符的一个小坑及解决》文章指出右移运算符处理负数时左侧补1导致死循环,与除法行为不同,强调需注意补码机制以正确统计二进制1的个数... 目录我遇到了这么一个www.chinasem.cn函数由此可以看到也很好理解总结我遇到了这么一个函数template<typename T>unsigned

C++统计函数执行时间的最佳实践

《C++统计函数执行时间的最佳实践》在软件开发过程中,性能分析是优化程序的重要环节,了解函数的执行时间分布对于识别性能瓶颈至关重要,本文将分享一个C++函数执行时间统计工具,希望对大家有所帮助... 目录前言工具特性核心设计1. 数据结构设计2. 单例模式管理器3. RAII自动计时使用方法基本用法高级用法

redis-sentinel基础概念及部署流程

《redis-sentinel基础概念及部署流程》RedisSentinel是Redis的高可用解决方案,通过监控主从节点、自动故障转移、通知机制及配置提供,实现集群故障恢复与服务持续可用,核心组件包... 目录一. 引言二. 核心功能三. 核心组件四. 故障转移流程五. 服务部署六. sentinel部署

从原理到实战解析Java Stream 的并行流性能优化

《从原理到实战解析JavaStream的并行流性能优化》本文给大家介绍JavaStream的并行流性能优化:从原理到实战的全攻略,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的... 目录一、并行流的核心原理与适用场景二、性能优化的核心策略1. 合理设置并行度:打破默认阈值2. 避免装箱

Python实战之SEO优化自动化工具开发指南

《Python实战之SEO优化自动化工具开发指南》在数字化营销时代,搜索引擎优化(SEO)已成为网站获取流量的重要手段,本文将带您使用Python开发一套完整的SEO自动化工具,需要的可以了解下... 目录前言项目概述技术栈选择核心模块实现1. 关键词研究模块2. 网站技术seo检测模块3. 内容优化分析模

深入解析C++ 中std::map内存管理

《深入解析C++中std::map内存管理》文章详解C++std::map内存管理,指出clear()仅删除元素可能不释放底层内存,建议用swap()与空map交换以彻底释放,针对指针类型需手动de... 目录1️、基本清空std::map2️、使用 swap 彻底释放内存3️、map 中存储指针类型的对象

Java实现复杂查询优化的7个技巧小结

《Java实现复杂查询优化的7个技巧小结》在Java项目中,复杂查询是开发者面临的“硬骨头”,本文将通过7个实战技巧,结合代码示例和性能对比,手把手教你如何让复杂查询变得优雅,大家可以根据需求进行选择... 目录一、复杂查询的痛点:为何你的代码“又臭又长”1.1冗余变量与中间状态1.2重复查询与性能陷阱1.

Python内存优化的实战技巧分享

《Python内存优化的实战技巧分享》Python作为一门解释型语言,虽然在开发效率上有着显著优势,但在执行效率方面往往被诟病,然而,通过合理的内存优化策略,我们可以让Python程序的运行速度提升3... 目录前言python内存管理机制引用计数机制垃圾回收机制内存泄漏的常见原因1. 循环引用2. 全局变

python 线程池顺序执行的方法实现

《python线程池顺序执行的方法实现》在Python中,线程池默认是并发执行任务的,但若需要实现任务的顺序执行,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋... 目录方案一:强制单线程(伪顺序执行)方案二:按提交顺序获取结果方案三:任务间依赖控制方案四:队列顺序消

使用Java读取本地文件并转换为MultipartFile对象的方法

《使用Java读取本地文件并转换为MultipartFile对象的方法》在许多JavaWeb应用中,我们经常会遇到将本地文件上传至服务器或其他系统的需求,在这种场景下,MultipartFile对象非... 目录1. 基本需求2. 自定义 MultipartFile 类3. 实现代码4. 代码解析5. 自定