C++基本函数的调用优化(构造、拷贝构造、赋值)
合理的函数可提升时间和空间的利用率
//Test1.h
#include<iostream>
using namespace std;
struct ST
{
private:
int a;
short b;
public:
ST(int a=0, short b=0):a(a),b(b)
{
this->a = a;
this->b = b;
cout<<"Object was Built. "<<this<<endl;
}
~ST()
{
cout<<"Object was Free. "<<this<<endl;
}
ST(const ST &t);
ST& operator=(const ST &t);
int Get_a();
};
ST::ST(const ST &t)
{
this->a = t.a;
this->b = t.b;
cout<<"Object was Copy. "<<this<<endl;
}
ST& ST::operator=(const ST &t)
{
cout<<"Assign:"<<this<<" = "<<&t<<endl;
if(this != &t)
{
this->a = t.a;
this->b = t.b;
}
return *this;
}
int ST::Get_a()
{
return this->a;
}
#include<iostream>
#include"Test1.h"
using namespace std;
ST fun(ST t)
{
int value = t.Get_a();
ST tmp(value);
return tmp;
}
void main()
{
ST t(12,13);
ST t1(t);
ST t2;
t2 = fun(t);
}
运行结果及分析①
1:对象t的构造和析构
2:对象t1的拷贝构造与析构
3:t2的构造与析构
ST fun(ST t)
{
int value = t.Get_a();
ST tmp(value);
return tmp;
}
4:fun()函数的参数为对象,调用拷贝构造函数创建临时对象,其在fun函数结束后被析构掉
5:fun()函数内使用ST实例化类tmp,其在函数结束后被析构掉
6:函数返回时调用拷贝构造函数创建一个临时对象,该对象在完成赋值后才被析构掉,所以6并不是tmp,这个临时对象可以生存到函数结束
运行结果及分析②
ST fun(ST t)
{
int value = t.Get_a();
return ST (value);
}
改变的代码仅有fun()函数变化
1,2,3,4与相同,不再赘述
这里的改变是不再创造有名对象,而是直接返回一个无名临时对象,将①的5,6合二为一,所以返回时仅需要使用构造函数构造一个无名临时对象,
在赋值后被析构掉。省掉了①中的拷贝构造。
运行结果及分析③
ST fun(ST &t)
{
int value = t.Get_a();
return ST(value);
}
与②相比,只改变了参数
1,2,3同上
当使用引用传递参数时,就不需要使用拷贝构造函数创建临时对象,将②的4省掉,③的4与②的4作用相同
运行结果及分析④
ST fun(ST &t)
{
int value = t.Get_a();
return ST(value);
}
void main()
{
ST t(12,13);
ST t1(t);
ST t2 = fun(t);
}
这里与③的不同是将主函数修改
主函数内st2从先使用构造函数初始化,再赋值,变成了直接使用赋值为其初始化
在前面我们知道
ST t(1,2);
ST st;
st = t;
和
ST t(1,2);
ST st = t;
实例化st 的不同是 前者先构造再拷贝构造,后者只需要拷贝构造。
所以,主函数使用了上例子后者的写法,可以直接省去t2的构造,直接拷贝构造,但是关键问题就在这里,
由于fun()函数返回的对象是一个无名的临时对象,所以编译器直接让其初始化t2,而省去拷贝构造的过程。
总结
1.从②我们可以知道,函数使用无名临时对象作为返回值,比在函数内创建有名临时对象更快,更节省空间,提升效率。
2.从③我们可以看出来在函数参数为对象时,使用引用可以省掉拷贝构造。
3.从④可以知道,在使用对象对对象进行赋值时,直接使用拷贝构造时最快的方法。这同时也说明了,无名的临时对象有时候可以被当作中间变量,而不需要从头进行拷贝构造工作。
C++基本函数的调用优化(构造、拷贝构造、赋值)的更多相关文章
- C++ //构造函数的分类及调用 //分类 // 按照参数分类 无参构造函数(默认构造) 有参构造函数 //按照类型分类 普通构造 拷贝构造
1 //构造函数的分类及调用 2 //分类 3 // 按照参数分类 无参构造函数(默认构造) 有参构造函数 4 //按照类型分类 普通构造 拷贝构造 5 6 #include <iostream ...
- C++编码优化之减少冗余拷贝或赋值
临时变量 目前遇到的一些产生临时变量的情况:函数实参.函数返回值.隐式类型转换.多余的拷贝 1. 函数实参 这点应该比较容易理解,函数参数,如果是实参传递的话,函数体里的修改并不会影响调用时传入的参数 ...
- c++拷贝构造和编译优化
#include <iostream> using namespace std; class MyClass { public: MyClass(); MyClass(int i); My ...
- C++ 构造函数、析构函数、拷贝构造、赋值运算符
之所以要把它们放在一起,是因为在使用C/C++类语言的时候,很容易混淆这几个概念(对Java来说完全没有这样的问题,表示Javaor完全没有压力). 先建立一个测试类(包含.h和.cpp) //~ P ...
- c++ 拷贝构造练习
#include<iostream> using namespace std; class Vector { private: int *dwp; int size; void clone ...
- C++ Primer笔记9_构造函数_拷贝构造(深拷贝与浅拷贝)
1.构造函数: >构造函数是一个特殊的.与类同名的成员函数,用于给每一个成员设置适当的初始值. >构造函数不能有返回值,函数名与类名同样. >缺省构造函数时,系统将自己主动调用该缺省 ...
- C++拷贝构造&操作符重载
头文件 DString.h如下 #ifndef __DSTRING_H #define __DSTRING_H #endif #include <stddef.h> class DStri ...
- C++对象的构造、析构与拷贝构造
今天下午在研究虚函数的时候遇到了一个问题,觉得很有意思,记录一下. 先看代码: class Base { public: Base(int value) { m_nValue = value; cou ...
- C++之旅:拷贝构造与友元
拷贝构造与友元 拷贝构造是在构造一个对象的时候将已有对象的属性拷贝给新的对象:友元可以让一个类的所有属性(主要是private)对特定的类开放 拷贝构造 如果没有复写拷贝构造函数,系统会帮我们默认生成 ...
随机推荐
- hdu 5095 Linearization of the kernel functions in SVM(模拟,分类清楚就行)
题意: INPUT: The input of the first line is an integer T, which is the number of test data (T<120). ...
- 力扣 - 剑指 Offer 58 - I. 翻转单词顺序
题目 剑指 Offer 58 - I. 翻转单词顺序 思路1 假如题目要求我们翻转字符串,那么我们可以从末尾往前开始遍历每一个字符,同时将每一个字符添加到临时空间,最后输出临时空间的数据就完成翻转了, ...
- nose在python2与python3中的包的自动发现用例的区别
最近在使用python3,同样装了nose,发现自动发现用例总是有问题,如下面的代码结婚 testcase |------ __init__.py |------ test_bb.py test_bb ...
- 痞子衡嵌入式:实测i.MXRT1010上的普通GPIO与高速GPIO极限翻转频率
大家好,我是痞子衡,是正经搞技术的痞子.今天痞子衡给大家介绍的是i.MXRT1010上的普通GPIO与高速GPIO极限翻转频率. 上一篇文章 <聊聊i.MXRT1xxx上的普通GPIO与高速GP ...
- 本地以sysdba 身份登录数据库实例时,报错ORA-01031 权限不足
在linux 操作系统的数据库服务器上,使用"sqlplus / as sysdba" 登录Oracle 10.2 数据库实例时,登录失败,显示ORA-01031: 权限不足. ...
- Vue组件传值prop验证方式
在Vue组件开发过程中,父组件要经常给子组件传递数据,在传递数据的过程中,子组件可以使用prop来接收父组件传递的值,同时呢,我们可以为组件的 prop 指定验证要求,例如你知道的这些类型.如果有一个 ...
- Chrome handless无界面浏览器的脚本操作
1.什么是Phantomjs (已经停止更新) 是一个无界面的浏览器 支持页面元素查找,js的执行等 由于不进行css和gui渲染,运行效率要比真实的浏览器要快很多 2.如何使用Phantomjs? ...
- 菜鸡的Java笔记 第十六 - java 引用传递
referenceDelivery 引用传递是整个java 的精髓,也是所有初学者最难学的地方 引用的本质:同一块堆内存可以被不同的栈内存所指向 下面通过三道程序来进行引用传 ...
- [luogu5180]支配树
对于有向图$G$和起点$s$,有以下定义和性质-- 为了方便,不妨假设$s$能到达$G$中所有点,并任意建立一棵以$s$为根的dfs树,以下节点比较默认均按照两点在这棵dfs树上的dfs序 支配点:$ ...
- [loj3014]独特的城市
约定:一棵树的深度定义为其中到根最远的点到根的距离 考虑节点$x$的答案: 任取一条直径,根据直径的性质,到$x$较远的直径端点一定是到$x$最远的点之一 由此,不难证明对于$x$独特的点,一定在$x ...