C++返回值优化RVO

返回值优化,是一种属于编译器的技术,它通过转换源代码和对象的创建来加快源代码的执行速度。RVO = return value optimization。

测试平台:STM32F103VG + Keil 5.15

背景:
我们有个MacAddress::ToArray

byte* MacAddress::ToArray() const
{
        return (byte*)&Value;
}

因为封装需要,打算返回字节数组类ByteArray的对象,于是有

ByteArray MacAddress::ToArray() const
{
        return ByteArray((byte*)&Value, 6);
}

调用代码

ByteArray bs = mac.ToArray();
bs.CopyTo(General_reg.SHAR);

按照我浅薄的C++知识理解,在ToArray内return 的时候,会产生一次对象拷贝,到临时对象。
然后在调用者那里的等号,产生一次拷贝构造。

实际上,编译烧写调试,查看反汇编

   358:         ByteArray bs = mac.ToArray();
0x0800595C 4629      MOV      r1,r5
0x0800595E A804      ADD      r0,sp,#0x10
0x08005960 F000FE92  BL.W     MacAddress::ToArray (0x08006688)
   359:         bs.CopyTo(General_reg.SHAR);
   360:
0x08005964 2300      MOVS     r3,#0x00
0x08005966 461A      MOV      r2,r3
0x08005968 F1040109  ADD      r1,r4,#0x09
0x0800596C A804      ADD      r0,sp,#0x10
0x0800596E F002FB8F  BL.W     Array::CopyTo (0x08008090)

直接分配内存,传入ToArray使用。ToArray之后,并没有见到所猜想的第二次拷贝构造。
下面看看ToArray的反汇编

0x08006688 B570      PUSH     {r4-r6,lr}
0x0800668A 4605      MOV      r5,r0
0x0800668C 460C      MOV      r4,r1
   481:         return ByteArray((byte*)&Value, 6);
0x0800668E 2206      MOVS     r2,#0x06
0x08006690 F1040108  ADD      r1,r4,#0x08
0x08006694 4628      MOV      r0,r5
0x08006696 F7FFFDEB  BL.W     _ZN9ByteArrayC2EPKhi (0x08006270)
0x0800669A 4605      MOV      r5,r0
   482: }
0x0800669C BD70      POP      {r4-r6,pc}

天哪!这里面只有一次构造函数,并不是猜想的那样,先构造本地变量,然后return再拷贝。
并且,这个构造函数的内存地址,正是外部传进去的那一个。

这个就是C++的RVO,返回值优化技术,没想到MDK也支持。

这个技能的获取,让我C++水平从30%提升到40%

时间: 2024-12-20 14:23:58

C++返回值优化RVO的相关文章

C++返回值优化

返回值优化(Return Value Optimization,简称RVO)是一种编译器优化机制:当函数需要返回一个对象的时候,如果自己创建一个临时对象用于返回,那么这个临时对象会消耗一个构造函数(Constructor)的调用.一个复制构造函数的调用(Copy Constructor)以及一个析构函数(Destructor)的调用的代价. 经过返回值优化,就可以将成本降低到一个构造函数的代价.这样就省去了一次拷贝构造函数的调用和依次析构函数的调用. 例子如下: class MyString {

转:C++中临时对象及返回值优化

http://www.cnblogs.com/xkfz007/articles/2506022.html 什么是临时对象? C++真正的临时对象是不可见的匿名对象,不会出现在你的源码中,但是程序在运行时确实生成了这样的对象. 通常出现在以下两种情况: (1)为了使函数调用成功而进行隐式类型转换的时候. 传递某对象给一个函数,而其类型与函数的形参类型不同时,如果可以通过隐式转化的话可以使函数调用成功,那么此时会通过构造函数生成一个临时对象,当函数返回时临时对象即自动销毁.如下例: //计算字符ch

[转] C++中临时对象及返回值优化

http://www.cnblogs.com/xkfz007/articles/2506022.html 什么是临时对象? C++真正的临时对象是不可见的匿名对象,不会出现在你的源码中,但是程序在运行时确实生成了这样的对象. 通常出现在以下两种情况: (1)为了使函数调用成功而进行隐式类型转换的时候. 传递某对象给一个函数,而其类型与函数的形参类型不同时,如果可以通过隐式转化的话可以使函数调用成功,那么此时会通过构造函数生成一个临时对象,当函数返回时临时对象即自动销毁.如下例: //计算字符ch

命名的返回值优化(Named Return Value optimization (NRVO))

命名的返回值优化: 针对返回一个局部的变量的优化,可以直接用返回的结果对象直接替代局部变量,从而减少了一个复制拷贝,从而提高效率. 比如 一个函数如下: X bar() { X xx; // .. 处理xx return xx; } 而在编译器看来则是如下的代码: // 此处的_result是一个在调用该函数时产生的一个临时对象 // , 然后将该对象传入,用以接受结果 void bar(X & __result) { X xx; // 调用xx的构造函数 xx.X::X(); // .. 处理

More Effective C++----(20)协助完成返回值优化

Item M20:协助完成返回值优化 一个返回对象的函数很难有较高的效率,因为传值返回会导致调用对象内的构造和析构函数(参见条款M19),这种调用是不能避免的.问题很简单:一个函数要么为了保证正确的行为而返回对象要么就不这么做.如果它返回了对象,就没有办法摆脱被返回的对象.就说到这. 考虑rational(有理数)类的成员函数operator*:(返回类型为const是为了防止连续对操作符的操作,一是返回的对象是一个临时对象,多个操作是在其临时对象上的操作,二是不符合内置类型的要求) class

参数返回值及NRV优化(named return value optimization)

C++11中的移动构造函数又把NRV优化翻出来了,都是采用临时中间值优化,两者不能共存. 参数传递如何实现? [实现模型1]引入临时对象,使用拷贝构造函数初始化.然后利用bitwise copy将其拷贝到x0的位置.比如: void foo( X x0 ); X xx; foo( xx ); 改写成 X __temp0; __temp0.X::X ( xx ); foo( __temp0 );还有一件事需要做,修改foo的声明,可以避免bit-wise copy的那一步. void foo( X

C++返回值为对象时复制构造函数不执行怎么破

先说点背景知识,调用复制构造函数的三种情况: 1.当用类一个对象去初始化另一个对象时. 2.如果函数形参是类对象. 3.如果函数返回值是类对象,函数执行完成返回调用时. 在辅导学生上机时,有同学第3点提出异议.有教材上的例题为证: #include <iostream> using namespace std; class Point //Point 类的定义 { public: Point(int xx=0, int yy=0) { x = xx; //构造函数,内联 y = yy; } P

异步上传文件并获得返回值(完全跨域)

http://blog.csdn.net/lrz1011/article/details/7913992 异步上传文件并获得返回值(完全跨域)AJAX可以进行数据的异步请求,但对于文件和跨域问题却束手无策. Jsonp可以进行跨域数据的异步请求,但同样不能使用于文件. <form>表单可以进行跨域数据和文件的上传,但却会使页面跳转. 那么如何同时实现“异步”+“跨域”+“文件”+“返回值”这几个特性呢?方法如下: 原理: 将<form>表单通过一个iframe来submit,也就是

js中父窗口获得模态窗口的返回值

<!--父窗口JS--> //查看审核详细 function viewFlow() {          var strBackValue= openWin('../VertifyView.aspx?&rn=' + Math.random()+'&View='+strValue, 630, 180);//这里的strBackValue就是模态窗口的返回值 alert(strBackValue) } //弹窗 function openWin(url, width, height