C++临时对象[转]

C++临时对象

2008-12-15 16:08:42|  分类: C技术 |  标签: |字号 订阅

临时对象是那些不是分配在堆里的匿名变量。

临时对象一般产生于两种场合:对函数调用时参数发生隐式类型转换;函数返回对象

隐式类型转换:

例如: 函数function(const string&),某个调用为:char* buf="xxx"; function(buf),在这时候,编译器会产生一个string类型的临时变量,该临时变量将buf转为string,然后传递给函数参数,直到函数返回,临时变量才被销毁。

值得注意的是:只有当参数类型为const,或者传值方式时,才能进行隐式转换。普通的传引用 方式不能进行隐式转换。这时因为,传入的对象地址实际上是临时对象的地址,如果在函数体内对参数进行了修改,改变的是临时变量的值,而用户传入的参数并没 有改变,这是与程序员的期望不一致。而传值与const类型的参数保证了函数体内的改变不影响到传入参数的值,因此可以这样使用。

 

 

/*此文是译者出于自娱翻译的GotW(Guru of the Week)系列文章第二篇,原文的版权是属于Hub Sutter(著名的C++专家,"Exceptional C++"的作者)。此文的翻译没有征得原作者的同意,只供学习讨论。——译者
*/

#2 临时对象
难度:5/10

不必要的临时对象常常导致代码冗余和执行效率低下。

问题:
    假设你正在看一段代码,代码中有如下一个函数。这个函数中至少有3处产生了不必要的临时对象。
    看看你能够找出几处,该怎样修改?
        
  string FindAddr( list<Employee> l, string name )
  {
    for( list<Employee>::iterator i = l.begin();
         i != l.end();
         i++ )
    {
      if( *i == name )
      {
        return (*i).addr;
      }
    }
    return "";
  }

答案:
    不管你信不信,就在这寥寥几行代码里出现了三处明显的不必要的临时对象,两处不太明显的临时对象,还有一处值得注意的地方。
   
   string FindAddr( list<Employee> l, string name )
                   ^^^^^^^1^^^^^^^^  ^^^^^2^^^^^
    1&2.参数应该是const引用类型.上述语句中的传值参数会导致liststring的拷贝操作,这可能是很耗时的。
           [准则]尽量使用const&来代替值拷贝传递参数。
   
   for( list<Employee>::iterator i = l.begin();
         i != l.end();
         i++ )
         ^3^
    3.此处比上两处稍难看出。此处如果用前自增操作代替后自增的将更有效率,因为对象的后自增操作需要对象执行自增操作并返回一个自增前原值的临时对象(译者:重载过前自增和后自增操作符的读者应该都清楚二者的区别)。注意这种情况对于象int这样的C++原始类型也是适用的。
           [准则]尽量使用前自增,避免使用后自增。
   
    if( *i == name )
           ^4
    4.虽然我们没有看到Employee类的定义,但是要使得上面的语句有效,这个类必须定义了到string类型转换操作符或者定义了以string类型为参数的构造函数。这两种情况都会产生一个临时对象,前者调用string的等值比较操作符(operator==),后者调用了Employee的等值比较操作符。(唯一不产生临时对象的情况就是stringEmployee类中重载了以对方为参数类型的等值比较操作符。)
           [准则]要小心隐藏在参数转换后面产生的临时对象。一个避免产生这种临时对象的解决办法就是用explicit修饰符对构造函数加以限制。
  
   return "";
         ^5
    5.此处产生了一个临时的空string对象。
    更好的方法是声明一个局部的string对象来存放返回值,并且最后以一条返回该对象的值的语句作为统一的返回出口。这将使得编译器可以在某些情况下采用返回值优化手段省略掉这个局部对象。比如下面的情形:调用者通过如下代码调用这个函数:
              string a = FindAddr( 1, "Harold" );
           [准则]遵循单个出口原则。决不要在同一个函数中存在多个返回语句。

[注意:在作了更多的性能测试以后,我并不完全赞成上述的原则。《Exception C++》中已经对此作了不同阐述。]

    string FindAddr( list< Employee> l, string name )
    ^^^*^^
    *.此处是一个题外话,但很值得注意。看起来好像简单地将函数返回值类型从 string类 型改为引用类型sting&就又可以避免产生一个临时对象,但这是错误的!如果你幸运的话,你的程序会在函数调用者使用返回的引用时就马上崩溃, 因为引用所指的局部对象已经不存在了。如果你不够幸运的话,你的程序看起来好像可以工作,但却不定期的崩溃,那将可能让你熬好几个晚上的长夜来调试找错。
           [准则]千万千万不要将一个局部对象的引用作为返回值。
         (注意:新闻组上有些人贴文正确的指出:可以通过声明一个静态对象,并在  没有查到对应雇员的地址时返回这个静态对象的引用,这样就可以把函数返回值改为引用类型而并不改变函数的语义。这同样也说明你在返回引用时必须了解所引用 对象的生命周期以保证返回的引用有效。)
   
    上述代码中还有一些可以优化的地方,比如可以避免调用end(),可以(或者说应该)使用一个const_iterator类型的迭代器。暂时不考虑这些,我们可以写出如下的较好的函数定义:
   string FindAddr( const list< Employee>& l, const string& name )
  {
    string addr;
    for( list< Employee>::const_iterator i = l.begin();
         i != l.end();
         ++i )
    {
      if( (*i). name == name )
      {
        addr = (*i).addr;
        break;
      }
    }
    return addr;
  }
----------
來源: http://blog.163.com/luoyanqing_mama/blog/static/69469482200811154842144/


比较另外的修改:
string FindAddr(const list< Employee>& l,
   const string& name)
{
string addr;
list< Employee>::const_iterator i=
   find(l.begin(),l.end(), name);
if(i != l.end())
{
    addr = (*i).addr;
}
return addr;
}

-------
来源: http://hi.baidu.com/kuroro2121/blog/item/a62c3a169e8a1bc0f6039ee0.html

来自 “ ITPUB博客 ” ,链接:http://blog.itpub.net/25306814/viewspace-703646/,如需转载,请注明出处,否则将追究法律责任。

转载于:http://blog.itpub.net/25306814/viewspace-703646/

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
回答: 在C++中,临时对象和匿名对象是两个不同的概念。 临时对象是在表达式中创建的对象,它们在表达式结束后会被销毁。临时对象通常用于函数返回值、函数参数传递和运算符重载等情况。例如,在例子1中,A(10)就是一个临时对象,它在调用myshow()函数后会被销毁。 匿名对象是没有被命名的对象,它们通常用于临时操作,不需要被引用或使用。匿名对象在创建后会立即被销毁。例如,在例子1中,A(11)和A(12)都是匿名对象,它们在被赋值给a和b后会被销毁。 临时对象和匿名对象的生命周期取决于它们的使用方式和上下文。在例子3中,Cat cc = Cat()创建了一个匿名对象,并将其赋值给cc对象,这个匿名对象的生命周期就变成了cc对象的生命周期。 总结起来,临时对象是在表达式中创建的对象,而匿名对象是没有被命名的对象。它们的生命周期取决于它们的使用方式和上下文。 #### 引用[.reference_title] - *1* *3* [C++匿名对象](https://blog.csdn.net/u014583317/article/details/108705360)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* [C++基础——匿名对象介绍、拷贝对象时的一些编译器优化](https://blog.csdn.net/weixin_69283129/article/details/127823416)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^insertT0,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值