Named Return Value Optimization (具名返回值优化)
本文转载自:http://blog.csdn.net/fanster28_/article/details/5006993
Named Return Value Optimization 具名返回值优化
问题描述 :
当一个函数返回一个对象的实例,一个临时对象被创建并通过拷贝构造函数传回其值。 C++ 标准允许省略拷贝构造函数的对象(即使会导致程序结果的不一致)。
前提是所有的路径返回相同的对象。
问题由来:
看如下的函数:
X bar() {
X xx;
//… 处理 xx
return xx;
}
如果看到了这篇文章,大概应该知道编译期面对这个函数会做如下的转化:
//Pseudo C++ Code
void bar( X &__result ) {
X xx;
xx.X::X(); //default constructor
// .. 处理 xx
__result.X::X( xx ); // copy constructor
return;
}
如上述 bar() 这样,所有的路径都返回相同的具名数值 (Named Value) ,因此编译器可以做自己的优化。以 __result 取代 Named Return Valued 。
//Pseudo C++ Code
void bar( X &__result ) {
__result.X::X(); //default constructor
// .. 处理 xx
return;
}
编译器呢?
B.Lippman 的 Inside the C++ Object Model 一书中如是说 :
NRV 优化如今被视为标准 C++ 编译器的一个义不容辞的优化操作。真的如此吗?
其实 Visual C++ 熬到了 8.0 才支持了 Named Return Value Optimization 。
实际的例子:
以下例子均是在 VS2010 beta2 中测得。
#include <ctime>
#include <memory>
#include <iostream>
using namespace std ;
class test {
friend test foo ( double );
public :
test () {
memset ( array , 0 , 100 *sizeof ( double ) );
cout << "test constructor " << '/n' ;
}
~test () {
cout << "test deconstructor " << '/n' ;
}
test ( const test &t );
private :
double array [ 100 ];
};
inline
test ::test ( const test &t ) {
memcpy ( this , &t , sizeof ( test ) );
cout << "test copy " << '/n' ;
}
test foo ( double val ) {
test local , local2 ;
local .array [ 0 ] = val ;
local .array [ 99 ] = val ;
if (val > -1 )
return local ;
}
int main() {
test t1 = foo( 2 );
return 0;
}
在没有 NRVO 优化的时候,编译优化选项关闭 (cl /Od) ,得到
test constructor
test constructor
test copy
test deconstructor
test deconstructor
test deconstructor
在打开 NRVO 优化时, (cl /O2) ,得到
test constructor
test constructor
test deconstructor
test deconstructor
可以看到,恰好是少了一次局部对象 xx 到 __result 的拷贝,以及析构。
注意的地方:
B.Lippman 在书中说到, NRVO 要由 Copy Constructor 激活,实际上不是的,至少 MS 在 VC中没有这样做。
注释掉上述拷贝构造函数, NRVO 依然可进行。实际上注释掉之后编译器会产生一个版本(bitwise copy ),可以从结果的两次构造,三次析构中看出来,少了的那次就是默认拷贝构造函数的调用。实行了优化后可见到只有两次析构的调用了,说明没有了默认拷贝构造函数的调用,编译器甚至不用产生它(需求时才产生)。
事实上,即使是没有定义拷贝构造函数和析构函数,而是有编译器产生(在需要时), NRVO依然会对程序的效率产生影响。如此能否说明 cfront 中以 是否存在拷贝构造作为 NRVO 的开关 存在争议?
首先注释调拷贝构造函数和析构函数 ,并在上述代码中增加:
friend test foo2 ( double );
test foo2 ( double val ) {
test local , local2 ;
local .array [ 0 ] = val ;
local .array [ 99 ] = val ;
if (val > -1 )
return local ;
return local2 ; //return 了不同的对象,编译器傻眼了 ^_^ ,不能 NVRO
}
注释调函数中的 cout 那些行,并 重写 main 函数如下:
int main () {
test *p = new test [100 ]; // 让 VC 不能把整个循环当常量表达式优化了
int cnt ;
clock_t begin = clock ();
for ( cnt = 0 ; cnt < 10000000 ; cnt ++ ) {
p [cnt %100 ] = foo ( double ( cnt ) );
}
clock_t end = clock ();
cout << "foo with NRVO used " << end - begin << "ms" << endl ;
begin = clock ();
for ( cnt = 0 ; cnt < 10000000 ; cnt ++ ) {
p [cnt %100 ] = foo2 ( double ( cnt ) );
}
end = clock ();
cout << "foo2 without NRVO used " << end - begin << "ms" << endl ;
delete []p ;
return 0 ;
}
在我这里测得结果如下:
foo with NRVO used 1480ms
foo2 without NRVO used 2631ms
可以看到没有 NVRO 时多的那一次 biwise copy 100*8bits 以及析构多花费的时间。
如果要知道更多编译器施行 NRVO 的限制,参考 这里 。
总结:
NRVO 曾经也饱受批评,有人认为编译器默默的做了这件事情,而程序员们并不知道是否做了,或者有些人不希望自己的程序被优化。
对于 NRVO 带来的副作用,可以考虑对象中有一个记录了其被拷贝次数的 static 变量,显然,NRVO 的应用可能会导致程序结果的不一致。
Named Return Value Optimization (具名返回值优化)相关推荐
- C++编程法则365条一天一条(358)copy elision(返回值优化NVO和具名返回值优化NRVO)
文章目录 强制编译器实现的优化 非强制实现优化 参考:https://en.cppreference.com/w/cpp/language/copy_elision Elision 是省略.删节或者忽 ...
- c++中返回值优化(RVO)和命名返回值优化(NRVO)介绍
RVO和NRVO介绍 前言 半年前就想写一篇关于RVO和NRVO的介绍,但碍于没什么时间去写博客.在跟身边人进行学术探讨的时候,会发现部分人可能尝到了编译器给它做返回值优化的好处,知道这段代码被优化了 ...
- 浅谈C++11标准中的复制省略(copy elision,也叫RVO返回值优化)
严正声明:本文系作者davidhopper原创,未经许可,不得转载. C++11以及之后的C++14.17标准均提出一项编译优化技术:复制省略(copy elision,也称复制消除),另外还有RVO ...
- 参数返回值及NRV优化(named return value optimization)
C++11中的移动构造函数又把NRV优化翻出来了,都是采用临时中间值优化,两者不能共存. 参数传递如何实现? [实现模型1]引入临时对象,使用拷贝构造函数初始化.然后利用bitwise copy将其拷 ...
- C++的返回值优化(RVO,Return Value Optimization)
前言 大家都知道"过早的优化是万恶之源"这句话,然而我相信其中的大多数人都不知道自己是不是在做过早的优化.我也无法准确的定义什么叫做"过早的优化",但我相信这& ...
- C++ 返回值优化 RVO
C++ 返回值优化 RVO 引子 返回值优化 RVO RVO 限制 参考 最近在调试代码时,发现拷贝和移动系列构造函数的调用和预期不太一样,经过查阅相关资料,发现是返回值优化(RVO)从中做梗.了解了 ...
- 什么是复制省略和返回值优化?
本文翻译自:What are copy elision and return value optimization? What is copy elision? 什么是复制省略? What is (n ...
- 提高C++性能的编程技术笔记:虚函数、返回值优化+测试代码
虚函数:在以下几个方面,虚函数可能会造成性能损失:构造函数必须初始化vptr(虚函数表):虚函数是通过指针间接调用的,所以必须先得到指向虚函数表的指针,然后再获得正确的函数偏移量:内联是在编译时决定的 ...
- C++高阶 返回值优化--RVO和NRVO介绍
RVO即返回值优化(return value optimize),可以少做一次拷贝构造. NRVO是具名返回值的意思,起初RVO技术仅支持匿名变量的优化,后期才加入对具名变量的优化. RVO: Big ...
- C++之RVO返回值优化
什么是RVO优化 RVO的全称是Return Value Optimization.RVO是一种编译器优化技术,可以把通过函数返回创建的临时对象给"去掉",然后可以达到少调用拷贝构 ...
最新文章
- sun m5000 snapshot
- 【干货】写给产品经理和设计师的用户体验知识
- vs2010 asp.net mysql,安装VS2010后,更改iis的asp.net版本 | 吴小强的博客
- 作者:赵晨(1980-),男,深圳般若计算机系统股份有限公司气象大数据应用负责人...
- 【数据结构与算法】伸展树的Java实现
- yum 更新内核报错 “Error: initscripts conflicts with centos-release-7-0.1406.el7.centos.2.3.x86_64的解决办法
- 【Java】synchronized与lock的区别
- javascript 弹出窗口中是否显示地址栏
- Swift - 使用导航条和导航条控制器来进行页面切换并传递数据
- 逆clarke变换_克拉克(CLARKE)及帕克(PARK)变换.pdf
- android app 尺寸,Android APP标注尺寸入门教程和app标注工具
- Mac谷歌浏览器无法下载的解决方案
- 人工智能 机器学习 数据挖掘 数据分析 算法大全
- 我热爱计算机作文450字,热爱音乐的我作文450字
- eigen冲突 sophus 安装_SVO-SLAM环境搭建指南
- ps4插html屏幕不亮光,万万没想到,居然是这个!PS4黑屏的原因终于找到了!
- h5广告与html5,那些刷爆朋友圈儿的H5广告大揭秘
- Amazon Dynamo论文解读 - Merkle Tree的使用
- 【吃瓜教程】随记第一篇
- 区分处理器的字,字长和汇编中的字,双字,四字