详解C++ 临时量与临时对象及程序的相关优化

一、临时量与临时对象

临时量:

  1. 内置类型生成的临时量是常量(临时量,寄存器带出来)。
  2. 自定义类型生成的临时量是变量 ,在内存中。
  3. 隐式生成生成的临时量是常量 ,显式生成生成的临时量是变量 。

临时对象:

临时对象是系统临时分配的对象,在没主动声明所需对象而又使用其功能时产生的

显示对象:出现类型名

隐式对象:不出现类型名

注意: 临时对象的生存周期只在本条语句,临时对象一旦被引用,它的生存周期就和引用相同。

对象如何生成?

先分配内存 在调用构造函数初始化对象的成员变量  产生对象对象析构了 对象就不存在了,对象的构造和析构是相反的。

重点:对象生成的顺序及调用的相关函数

class Test
{
public:
  Test(int a=5, int b=5):ma(a), mb(b)
  {cout<<"Test(int, int)"<<endl;}
  ~Test()
  {cout<<"~Test()"<<endl;}
  Test(const Test &src):ma(src.ma), mb(src.mb)
  {cout<<"Test(const Test&)"<<endl;}
  void operator=(const Test &src)
  {ma = src.ma; mb = src.mb; cout<<"operator="<<endl;}
private:
  int ma;
  int mb;
};
Test t1(10, 10);
int main()
{
  Test t2(20, 20);
  Test t3=t2;
  static Test t4 = Test(30, 30);
  t2 = Test(40, 40);
  t2 = (Test)(50, 50);
  t2 = 60;
  Test *p1 = new Test(70, 70);
  Test *p2 = new Test[2];
  Test *p3 = &Test(80, 80);
  Test &p4 = Test(90, 90);
  delete p1;
  delete []p2;
}
Test t5(100, 100);

Test(int) // Test t1(10,10); 构造t1
Test(int) // Test t5(10,10); 构造t5
Test(int) // Test t2(20 ,20); 构造t2
Test(const Test &) // Test t3 = t2; t2拷贝构造t3
Test(int)	// static Test t4 = Test(30,30); 构造t4
Test(int)	// t2 = Test(40,40); 构造临时对象
Test& operator=(const Test &) // t2 = Test(40,40); 临时对象赋值给t2
~Test()		// t2 = Test(40,40); 析构临时对象
 Test(int)	// t2 = (Test)(40,50); 构造临时对象  逗号表达式 t2 = (Test)(50)  50 , 5
Test& operator=(const Test &) // t2 = (Test)(40,50); 临时对象赋值给t2
 ~Test()		// t2 = (Test)(40,50);析构临时对象
Test(int)	// t2 = 60; 构造临时对象 相当于 t2 = (Test)60;
Test& operator=(const Test &) // t2 = 60; 临时对象赋值给t2
~Test()		// t2 = 60; 析构临时对象
Test(int)	// Test *p1 = new Test; 构造 Test
Test(int)	// Test *p2 = new Test[2]; 构造 Test
Test(int)	// Test *p2 = new Test[2]; 构造 Test
Test(int)	// Test *p3 = &Test(50,50); 构造临时对象
~Test()		// Test *p3 = &Test(50,50); 析构临时对象
Test(int)	// Test &p4 = Test(50,50); 构造临时对象
~Test()		// 析构p1
~Test()		// 析构p2
 ~Test()		// 析构p2
 ~Test()		// 析构p4指向的临时对象
~Test()		// 析构t3
 ~Test()		// 析构t2
 ~Test()		// 析构t4
 ~Test()		// 析构t5
 ~Test()		// 析构t1
!

二、程序优化

  1. 1.函数调用传对象时,按对象引用来传递,会少两个函数
  2. 2.函数返回对象的时候,应该返回一个临时对象,不要先定义,再返回
  3. 3.调用返回对象的函数时,应该以初始化的方式调用,不要以赋值的方式调用
class Test
{
public:
	Test(int data = 100) :ma(data)
	{
		cout << "Test(int)" << endl;
	}
	~Test()
	{
		cout << "~Test()" << endl;
	}
	Test(const Test &src) :ma(src.ma)
	{
		cout << "Test(const Test&)" << endl;
	}
	Test& operator=(const Test &src)
	{
		cout << "operator=" << endl;
		ma = src.ma;
		return *this;
	}
	int getData() { return ma; }
private:
	int ma;
};

Test GetTestObject(Test t)
{
	int value = t.getData();
	Test tmp(value);
	return tmp;
	//return Test(value);
}
int main()
{
	Test t1;
	Test t2;
	t2 = GetTestObject(t1);
	cout << t2.getData() << endl;

	return 0;
}

程序分析

// 构造t1
Test(int) 

// 构造t2
Test(int) 

// GetTestObject(t1)实参t1通过值传递给Test GetTestObject(Test t) 形参 t ,调用拷贝构造
Test(const Test &) 

//Test tmp(value); 构造对象tmp
Test(int)

//return tmp; 将返回值tmp拷贝构造 main函数栈栈上的临时对象
Test(const Test&)

// 析构tmp
~Test()

// 析构形参 t
~Test()

// t2 = GetTestObject(t1); 临时对象调用赋值函数赋值给t2
operator=

// 析构临时对象
~Test()

// 打印 ma
100

// 析构t2
~Test()

// 析构t1
~Test()

优化1:函数调用传对象时,按对象引用来传递,会少两个函数

Test GetTestObject(Test &t)
{
	int value = t.getData();
	Test tmp(value);
	return tmp;
}

优化2:函数返回对象的时候,应该返回一个临时对象,不要先定义,再返回

Test GetTestObject(Test &t)
{
	int value = t.getData();
	/*Test tmp(value);
	return tmp;*/
	return Test(value);
}

优化3:调用返回对象的函数时,应该以初始化的方式调用,不要以赋值的方式调用

int main()
{
	Test t1;
	Test t2 = GetTestObject(t1);
	//t2 = GetTestObject(t1);
	cout << t2.getData() << endl;

	return 0;
}

以上所述是小编给大家介绍的C++ 临时量与临时对象及程序的相关优化详解整合,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对我们网站的支持!

(0)

相关推荐

  • 深入c++中临时对象的析构时机的详解

    c++中,临时对象一旦不需要,就会调用析构函数,释放其占有的资源:而具名对象则是与创建的顺序相反,依次调用析构函数. c++源码: 复制代码 代码如下: class X  {public:   int i;   int j;   ~X() {}   X() {} }; int main() {    X x1;    X();    x1.i = 1;    X x2; } 对应的汇编码: 复制代码 代码如下: _main    PROC ; 11   : int main() { push  

  • c++ 临时对象的来源

    首先看下面一端代码: 复制代码 代码如下: #include <iostream> void swap( int &a,int &b) {     int temp;     temp=a;     a=b;     b=temp; } int main(int argc,char** argv) {     int a=1,b=2;     swap(a,b);     std::cout<<a<<"-----"<<b&

  • 详解C++ 临时量与临时对象及程序的相关优化

    一.临时量与临时对象 临时量: 内置类型生成的临时量是常量(临时量,寄存器带出来). 自定义类型生成的临时量是变量 ,在内存中. 隐式生成生成的临时量是常量 ,显式生成生成的临时量是变量 . 临时对象: 临时对象是系统临时分配的对象,在没主动声明所需对象而又使用其功能时产生的 显示对象:出现类型名 隐式对象:不出现类型名 注意: 临时对象的生存周期只在本条语句,临时对象一旦被引用,它的生存周期就和引用相同. 对象如何生成? 先分配内存 在调用构造函数初始化对象的成员变量  产生对象对象析构了 对

  • 详解处理Java中的大对象的方法

    目录 String中的substring 集合大对象扩容 保持合适的对象粒度 Bitmap 把对象变小 数据的冷热分离 数据双写 写入 MQ 分发 使用 Binlog 同步 思维发散 小结 本文我们将讲解一下对于“大对象”的优化.这里的“大对象”,是一个泛化概念,它可能存放在 JVM 中,也可能正在网络上传输,也可能存在于数据库中. 那么为什么大对象会影响我们的应用性能呢? 第一,大对象占用的资源多,垃圾回收器要花一部分精力去对它进行回收: 第二,大对象在不同的设备之间交换,会耗费网络流量,以及

  • 详解Java中的不可变对象

    不可变对象想必大部分朋友都不陌生,大家在平时写代码的过程中100%会使用到不可变对象,比如最常见的String对象.包装器对象等,那么到底为何Java语言要这么设计,真正意图和考虑点是什么?可能一些朋友没有细想过这些问题,今天我们就来聊聊跟不可变对象有关的话题. 一.什么是不可变对象 下面是<Effective Java>这本书对于不可变对象的定义: 不可变对象(Immutable Object):对象一旦被创建后,对象所有的状态及属性在其生命周期内不会发生任何变化. 从不可变对象的定义来看,

  • 一文详解Java如何创建和销毁对象

    目录 一.介绍 二.实例构造(Instance Construction) 2.1 隐式(implicitly)构造器 2.2 无参构造器(Constructors without Arguments) 2.3 有参构造器(Constructors with Arguments) 2.4 初始化块(Initialization Blocks) 2.5 构造保障(Construction guarantee) 2.6 可见性(Visibility) 2.7 垃圾回收(Garbage collect

  • 详解JavaScript中的函数、对象

    JS中的函数声明方式 方式一 function 函数名(){ 函数体 } 方式二 var 函数名=function(){ 函数体 } 方式三 var 函数名=new Function("函数体"): 执行方式 函数名(); JS中的对象 类似Java中的一些系统预设好的类 日期对象 function testDate(){ var date=new Date(); //本月中的第几天 document.write(date.getDate()+"<br />&qu

  • 详解Java在redis中进行对象的缓存

    Java在redis中进行对象的缓存一般有两种方法,这里介绍序列化的方法,个人感觉比较方便,不需要转来转去. 一.首先,在存储的对象上实现序列化的接口 package com.cy.example.entity.system; import java.util.List; import com.baomidou.mybatisplus.annotations.TableField; import com.baomidou.mybatisplus.annotations.TableName; im

  • Python函数基础实例详解【函数嵌套,命名空间,函数对象,闭包函数等】

    本文实例讲述了Python函数基础用法.分享给大家供大家参考,具体如下: 一.什么是命名关键字参数? 格式: 在*后面参数都是命名关键字参数. 特点: 1.约束函数的调用者必须按照Kye=value的形式传值. 2.约束函数的调用者必须用我们指定的Key名. def auth(*args,name,pwd): print(name,pwd) auth(pwd='213',name='egon') def register(name,age): print(type(name),type(age)

  • 详解Android轻量型数据库SQLite

    数据库是Android存储方案的核心,在Andorid中SQLite非常轻量,而且执行sql语句甚至比MySQL还要快. SQLiteDatabase 是 Android 中操作数据库的核心类之一,使用SQLiteDatabase可以打开数据库,也可以对数据库进行操作,然而,为了数据库升级以及使用更加方便,我们常用SQLiteOpenHelper的子类来完成创建,打开数据库的操作. SQLiteOpenHelper是一个抽象类,在该类中有下面两个必须实现的方法: public void onCr

  • 详解Ruby中的代码块对象Proc

    Proc对象 Proc是由块转换来的对象.创建一个Proc共有四种方法,分别是: 示例代码 # 法一 inc = Proc.new { | x | x + 1} inc.call(2) #=> 3 # 法二 inc = lambda {| x | x + 1 } inc.call(2) #=> 3 # 法三 inc = ->(x) { x + 1} inc.call(2) #=> 3 # 法四 inc = proc {|x| x + 1 } inc.call(2) #=> 3

  • 详解SQL Server数据库架构和对象、定义数据完整性

    前言 本节我们继续SQL之旅,本节我们如题来讲讲一些基本知识以及需要注意的地方,若有不妥之处,还望指出,简短的内容,深入的理解. 数据库架构和对象 数据库包含架构,而架构又包含对象,架构可以看做是表.视图.存储过程等对象的容器.架构是一个命名空间,它被用做对象名称的前缀,比如在Cnblogs的架构中有一个名称为Blogs的表,此时我们用架构式限定式名称(即两部分式对象名称)所以Blogs表示为Cnblogs.Blogs.如果我们引用对象时省略了架构名称,SQL Server将会检查对象是否存在用

随机推荐