`
美丽的小岛
  • 浏览: 312292 次
  • 性别: Icon_minigender_1
  • 来自: 大连
社区版块
存档分类
最新评论

C++临时对象

    博客分类:
  • c++
 
阅读更多

C++中有这样一种对象:它在代码中看不到,但是确实存在。它就是临时对象---由编译器定义的一个没有命名的非堆对象(non-heap object)。为什么研究临时对象?主要是为了提高程序的性能以及效率,因为临时对象的构造与析构对系统性能而言绝不是微小的影响,所以我们应该去了解它们,知道它们如何造成,从而尽可能去避免它们。

 

临时对象通常产生于以下4种情况:

 

  1. 类型装换
  2. 按值传递
  3. 按值返回
  4. 对象定义

 

下面我们逐一看看:

1、类型转换:它通常是为了让函数调用成功而产生临时对象。发生于 “传递某对象给一个函数,而其类型与它即将绑定上去的参数类型不同” 的时候。

例如:

void test(const string& str);

char buffer[] = "buffer";

test(buffer); // 此时发生类型转换

 此时,编译器会帮你进行类型转换:它产生一个类型为string的临时对象,该对象以buffer为参数调用string constructor。当test函数返回时,此临时对象会被自动销毁。

 

注意:对于引用(reference)参数而言,只有当对象被传递给一个reference-to-const参数时,转换才发生。如果对象传递给一个reference-to-non-const对象,不会发生转换。

例如:

void upper(string& str);

char lower[] = "lower";

upper(lower); // 此时不能转换,编译出错

 此时如果编译器对reference-to-non-const对象进行了类型转换,那么将会允许临时对象的值被修改。而这和程序员的期望是不一致的。试想,在上面的代码中,如果编译器允许upper运行,将lower中的值转换为大写,但是这是对临时对象而言的,char lower[]的值还是“lower”,这和你的期望一致吗?

 

有时候,这种隐式类型转换不是我们期望的,那么我们可以通过声明constructor为explicit来实现。explicit告诉编译器,我们反对将constructor用于类型转换。

例如:

explicit string(const char*);

 2、按值传递:这通常也是为了让函数调用成功而产生临时对象。当按值传递对象时,实参对形参的初始化与T formalArg = actualArg的形式等价。

 

例如:

void test(T formalArg);

T actualArg;
test(actualArg);

 此时编译器产生的伪码为:

T _temp;

_temp.T::T(acutalArg); // 通过拷贝构造函数生成_temp
g(_temp);  // 按引用传递_temp
_temp.T::~T(); // 析构_temp

 因为存在局部参数formalArg,test()的调用栈中将存在formalArg的占位符。编译器必须复制对象actualArg的内容到formalArg的占位符中。所以,此时编译器生成了临时对象。

 

3、按值返回:如果函数是按值返回的,那么编译器很可能为之产生临时对象。

例如:

class Integer {
public:
  friend Integer operator+(const Integer& a, const Integer& b);
  
  Integer(int val=0): value(val) {
  }
  
  Integer(const Integer& rhs): value(rhs.value) {
  }
  
  Integer& operator=(const Integer& rhs);
  
  ~Integer() {
  }
  
private:
  int value;  
};

Integer operator+(const Integer& a, const Integer& b) {
  Integer retVal;
  
  retVal.value = a.value + b.value;
  
  return retVal;
}

Integer c1, c2, c3;
c3 = c1 + c2;

 编译器生成的伪代码:

struct Integer _tempResult; // 表示占位符,不调用构造函数
operator+(_tempResult, c1, c2); // 所有参数按引用传递
c3 = _tempResult; // operator=函数执行

Integer operator+(const Integer& _tempResult, const Integer& a, const Integer& b) {
  struct Integer retVal;
  retVal.Integer::Integer(); // Integer(int val=0)执行
  
  retVal.value = a.value + b.value;
  
  _tempResult.Integer::Integer(retVal); // 拷贝构造函数Integer(const Integer& rhs)执行,生成临时对象。
  
  retVal.Integer::~Integer(); // 析构函数执行
  
  return;
}
  
  return retVal;
}

 如果对operator+进行返回值优化(RVO:Return Value Optimization),那么临时对象将不会产生。

 

例如:

Integer operator+(const Integer& a, const Integer& b) {  
  return Integer(a.value + b.value);
}

 编译器生成的伪代码:

Integer operator+(const Integer& _tempResult, const Integer& a, const Integer& b) {
  _tempResult.Integer::Integer(); // Integer(int val=0)执行
  _tempResult.value = a.value + b.value;
  
  return;
}

 对照上面的版本,我们可以看出临时对象retVal消除了。

 

4、对象定义:

例如:

Integer i1(100); // 编译器肯定不会生成临时对象
Integer i2 = Integer(100); // 编译器可能生成临时对象
Integer i3 = 100; // 编译器可能生成临时对象

 然而,实际上大多数的编译器都会通过优化省去临时对象,所以这里的初始化形式基本上在效率上都是相同的。

 

 

备注:

临时对象的生命期:按照C++标准的说法,临时对象的摧毁,是对完整表达式求值过程中的最后一个步骤。该完整表达式照成了临时对象的产生。

完整表达式通常是指包含临时对象表达式的最外围的那个。例如:

((objA >1024)&&(objB <1024) ) ? (objA - objB) :(objB-objA)

这个表达式中一共含有5个表达式,最外围的表达式是?。任何一个子表达式所产生的任何一个临时对象,都应该在完整表达式被求值完成后,才可以销毁。

临时对象的生命周期规则有2个例外

1、在表达式被用来初始化一个object时。例如:

 

[cpp] view plaincopy
 
  1. String progName("test");  
  2. String progVersion("ver-1.0");  
  3. String progNameVersion = progName + progVersion  


如果progName + progVersion产生的临时对象在表达式求值结束后就析构,那么progNameVersion就无法产生。所以,C++标准规定:含有表达式执行结果的临时对象,应该保留到object的初始化操作完成为止。

 

小心这种情况:

 

[cpp] view plaincopy
 
  1. const char* progNameVersion = progName + progVersion  


这个初始化操作是一定会失败的。编译器产生的伪码为:

 

 

[cpp] view plaincopy
 
  1. String _temp;  
  2. operator+(_temp, progName, progVersion);  
  3. progNameVersion = _temp.String::operator char*();  
  4. _temp.String::~String();  


2、当一个临时对象被一个reference绑定时。例如:

 

 

[cpp] view plaincopy
 
  1. const String& name = "C++";  


编译器产生的伪码为:

 

 

[cpp] view plaincopy
 
  1. String _temp;  
  2. temp.String::String("C++");  
  3. const String& name = _temp;  


针对这种情况,C++标准上是这样说的:如果一个临时对象被绑定于一个reference,对象将保留,直到被初始化的reference的生命结束,或直到临时对象的生命范围结束-----看哪种情况先到达而定。

 

 

参考书籍:

1、《深度探索:C++对象模型》

2、《提高C++性能的编程技术》

3、《Effective C++》

4、《more effective C++》

5、《C++语言的设计和演化》

http://blog.csdn.net/imyfriend/article/details/12886577

分享到:
评论

相关推荐

    C++l临时对象

    ### C++中的临时对象 #### 一、引言 在C++编程中,临时对象是一个非常重要但又常常被忽视的概念。这些对象虽然不直接出现在源代码中,但在编译器生成的目标代码中却扮演着至关重要的角色。了解临时对象如何产生、...

    C++临时对象详解[收集].pdf

    C++中的临时对象是指在表达式中为了完成特定任务而临时创建的对象,它们在程序运行过程中通常是不可见的,并且生命期很短。临时对象的创建可能会带来性能上的开销,尤其是在涉及拷贝构造函数或者多态性时。在本文中...

    c++ 临时对象的来源

    《C++临时对象的来源与理解》 在C++编程中,临时对象是一个不显式声明,但在程序执行过程中产生的对象。这些对象虽然在代码中不可见,但它们在特定情况下是必要的,尤其是在类型转换、函数调用和返回值优化(RVO)...

    详解C++ 中的临时对象

    在C++编程语言中,临时对象(Temporary Object)是指在表达式中为了完成特定操作而临时创建的对象。这些对象通常不被程序员直接命名,而是由编译器在幕后生成并管理。临时对象主要出现在以下几个场景: 1. **无名...

    C++临时性对象的生命周期详细解析

    此外,C++11引入了移动语义(Move Semantics),通过右值引用(rvalue reference)进一步优化了临时对象的处理,允许更高效地转移资源,而不是拷贝。在编写C++代码时,理解临时性对象的生命周期能够帮助我们编写出...

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

    在C++编程语言中,临时对象(Temporary Object)是指在表达式中为了完成某个操作而临时创建的对象,它们没有名称并且通常在表达式完成后立即销毁。临时对象的析构时机是一个重要的概念,因为它直接影响到程序的内存...

    深度探索C++对象模型 PDF中文清晰版

    《深度探索C++对象模型》专注于C++面向对象程序设计的底层机制,包括结构式语意、临时性对象的生成、封装、继承,以及虚拟——虚拟函数和虚拟继承。这本书让你知道:一旦你能够了解底层实现模型,你的程序代码将获得...

    小问题大思考之C++ 临时对象

    它是临时对象—由编译器定义的一个没有命名的非堆对象(non-heap object)。为什么研究临时对象?主要是为了提高程序的性能以及效率,因为临时对象的构造与析构对系统性能而言绝不是微小的影响,所以我们应该去了解...

    深入解析C++中的临时变量

    深入解析C++中的临时变量:理解其本质与作用 在C++编程中,临时变量扮演着至关重要的角色,尽管它们通常在代码中“隐身”,但它们的存在极大地影响着程序的性能和行为。本文将深入探讨C++中的临时变量,从操作系统...

    深度探索C++对象模型(中文版&英文版)

    本书探讨了大量的C++面向对象程序设计的底层运作机制,包括构造函数,函数,临时对象,继承,虚拟,模板的实例化,异常处理,运行期类型识别等,另外还介绍了一些在实现C++对象模型过程中做出的权衡折衷.喜欢刨根问底的C++...

    C++单向按值传递(产生临时对象) 双向按引用传递(不产生临时对象不占任何内存).rar

    在C++中,当一个对象作为参数按值传递时,会创建该对象的一个临时副本。这个过程可能导致性能上的开销,特别是在传递大型数据结构时,因为需要复制整个对象的内存空间。 例如: ```cpp void swap(int a, int b) { ...

    深入探索C++对象模型-总结笔记.pdf

    本书专注于C++面向对象程序设计的底层机制,包括结构式语意、临时性对象的生成、封装、继承,以及虚拟——虚拟函数和虚拟继承。这本书让你知道:一旦你能够了解底层实现模型,你的程序代码将获得多么大的效率。...

    深度探索C++对象模型读书笔记

    - **临时对象的管理**:讨论临时对象的生命周期及其如何被正确处理。 #### 结语 通过以上内容,我们可以看出C++对象模型是一个庞大且复杂的主题。理解这些基础知识不仅可以帮助我们编写出更高效、更安全的代码,还...

    深入C++对象模型(英文版)

    本书探讨了大量的C++面向对象程序设计的底层运作机制,包括构造函数,函数,临时对象,继承,虚拟,模板的实例化,异常处理,运行期类型识别等,另外还介绍了一些在实现C++对象模型过程中做出的权衡折衷.喜欢刨根问底的C++...

    深度探索C++对象模型2012版.rar

    本书专注于C++面向对象程序设计的底层机制,包括结构式语意、临时性对象的生成、封装、继承,以及虚拟——虚拟函数和虚拟继承。这本书让你知道:一旦你能够了解底层实现模型,你的程序代码将获得多么大的效率。...

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

    在C++编程中,临时量和临时对象是两个重要的概念,它们在程序执行过程中起着关键的作用。本文将深入探讨这两个概念以及相关的程序优化策略。 首先,临时量是指在表达式中生成的、用于临时存储结果的量。对于内置...

    c++手册 对象的演化

    构造器初值列表用于在构造函数中初始化成员变量,避免了隐式类型转换和不必要的临时对象创建。掌握它的用法可以提高代码的效率和清晰度。 七、对象池与对象工厂 对于大量短生命周期的对象,使用对象池可以减少频繁...

Global site tag (gtag.js) - Google Analytics