关于C++右值引用的参考文档里面有明确提到,右值引用可以延长临时变量的周期。如:

std::string&& r3 = s1 + s1; // okay: rvalue reference extends lifetime

看到这里的时候,Binfun有点崩溃,就这就能延长生命周期?这个和以下的这样的命令有啥本质的区别吗?

std::string r3 = s1 + s1

所以Binfun写了一段小代码来测试一下右值引用的延长生命周期的特性,如:

#include <stdio.h>
#include <utility>//std::move class result {
public:
int val;
result() { printf("constructor() [%p]\n", this); }
result(result& r): val(r.val) { printf("copying from [%p] to [%p]\n", &r, this); }
result(result&& r): val(r.val) { printf("moving from [%p] to [%p]\n", &r, this); }
result(int i): val(i) { printf("constructor(%d) [%p]\n", val, this); }
~result() { printf("destructor() [%p]\n", this); }
}; result process(int i)
{
printf("In process function\n");
return result(i);
} int main()
{
printf("---step1---\n");
result s1 = process(1);
printf("---step2---\n");
result &&s2 = process(2); printf("---vals:---\n");
printf("s1 addr:[%p], val:[%d]\n", &s1, s1.val);
printf("s2 addr:[%p], val:[%d]\n", &s2, s2.val);
}

然后Binfun自信满满地敲了编译并执行命令:

g++ new_move.cpp -std=c++11 -O2 && ./a.out

看到打印的时候Binfun再一次崩溃了:

---step1---
In process function
constructor(1) [0x7ffd94c8aca0]
---step2---
In process function
constructor(2) [0x7ffd94c8acb0]
---vals:---
s1 addr:[0x7ffd94c8aca0], val:[1]
s2 addr:[0x7ffd94c8acb0], val:[2]
destructor() [0x7ffd94c8acb0]
destructor() [0x7ffd94c8aca0]

这……没有任何区别啊,C++国际标准委员会逗我玩呢?

RVO和右值引用

其实是有区别的,先听我解释一下RVO这个概念:返回值优化

返回值优化(Return value optimization,缩写为RVO)是C++的一项编译优化技术。即删除保持函数返回值的临时对象。这可能会省略多次复制构造函数

在调用process函数的时候竟然没有临时变量产生(可以看到构造函数只运行了一次),那应该是被RVO了。既然是编译优化技术,那么应该有编译选项关闭,RVO优化在C++里面也叫copy_elision(复制消除)优化。使用以下命令即可取消RVO:

g++ new_move.cpp -std=c++11 -fno-elide-constructors -O2 && ./a.out

编译后打印如下:

---step1---
In process function
constructor(1) [0x7ffe849b8a70]
moving from [0x7ffe849b8a70] to [0x7ffe849b8ab0]
destructor() [0x7ffe849b8a70]
moving from [0x7ffe849b8ab0] to [0x7ffe849b8aa0]
destructor() [0x7ffe849b8ab0]
---step2---
In process function
constructor(2) [0x7ffe849b8a70]
moving from [0x7ffe849b8a70] to [0x7ffe849b8ab0]
destructor() [0x7ffe849b8a70]
---vals:---
s1 addr:[0x7ffe849b8aa0], val:[1]
s2 addr:[0x7ffe849b8ab0], val:[2]
destructor() [0x7ffe849b8ab0]
destructor() [0x7ffe849b8aa0]

可以看到在step1中调用process函数的时候,构造产生了一个变量(地址为0x7ffe849b8a70),然后函数返回时将这个变量移动构造到了另一个临时变量(地址为0x7ffe849b8ab0),接着赋值给s2(地址为0x7ffe849b8aa0)时再一次调用了移动构造函数。

之所以以上调用的都是移动构造,这是因为编译器识别出这些变量都是“将亡值”,也就是说编译器知道这个变量接下来都会离开它的作用域,即将会被析构掉,此时认定它是一个右值&&,所以也就调用的是移动构造函数。打印中也体现了这一点,0x7ffe849b8a70和0x7ffe849b8ab0被move constructor之后立马就被析构掉了。

而step2就不一样了,我们看到了移动构造函数只被调用了一次。而这一次0x7ffe849b8ab0并没有被析构掉,这一次它被保留了下来,它的生命周期被延长了,直到main函数结束时它才会析构掉。

以上可以看到右值引用的确可以延长右值变量的生命周期。当然尽管在RVO的光环下,只需要构造一次就已经到位了,就没必要去延长生命周期了-。-|||

std::move和右值引用的坑

另外Binfun进一步理解移动语义的时候,发现了一个坑,希望大家关注一下。这个坑会导致奇怪的问题发生……也会导致很隐蔽的bug……

在说这个例子的时候,我们先介绍一下std::move(懂的可以略过-。-)

文章最下面的参考链接里面的文章有一段写得特别棒,如下:

关于move函数内部到底是怎么实现的,其实std::move函数并不“移动”,它仅仅进行了类型转换。下面给出一个简化版本的std::move:

template <typename T>
typename remove_reference<T>::type&& move(T&& param)
{
using ReturnType = typename remove_reference<T>::type&&;
return static_cast<ReturnType>(param);
}

代码很短,但是估计很难懂。首先看一下函数的返回类型,remove_reference在头文件中,remove_reference有一个成员type,是T去除引用后的类型,所以remove_reference::type&&一定是右值引用,对于返回类型为右值的函数其返回值是一个右值(准确地说是xvalue)。所以,知道了std::move函数的返回值是一个右值。然后,我们看一下函数的参数,使用的是通用引用类型(&&),意味者其可以接收左值,也可以接收右值。但是不管怎么推导,ReturnType的类型一定是右值引用,最后std::move函数只是简单地调用static_cast将参数转化为右值引用。

还是同样的result类和同样的process函数,我们修改一下main函数为:

int main()
{
printf("---step1---\n");
result &&s1 = std::move(process(1));
printf("---step2---\n");
process(2); printf("---vals:---\n");
printf("s1 addr:[%p], val:[%d]\n", &s1, s1.val);
}

猜猜最后打印中s1.val的值是1还是2?

g++ new_move.cpp -std=c++11 -O2 && ./a.out

以上的编译选项的打印如下:

---step1---
In process function
constructor(1) [0x7ffdf1352db0]
destructor() [0x7ffdf1352db0]
---step2---
In process function
constructor(2) [0x7ffdf1352db0]
destructor() [0x7ffdf1352db0]
---vals:---
s1 addr:[0x7ffdf1352db0], val:[2]

g++ new_move.cpp -std=c++11 -fno-elide-constructors -O2 && ./a.out

以上的编译选项,打印结果如下:

---step1---
In process function
constructor(1) [0x7ffe7d59f350]
moving from [0x7ffe7d59f350] to [0x7ffe7d59f380]
destructor() [0x7ffe7d59f350]
destructor() [0x7ffe7d59f380]
---step2---
In process function
constructor(2) [0x7ffe7d59f350]
moving from [0x7ffe7d59f350] to [0x7ffe7d59f380]
destructor() [0x7ffe7d59f350]
destructor() [0x7ffe7d59f380]
---vals:---
s1 addr:[0x7ffe7d59f380], val:[2]

很可惜都是2,不管有没有RVO优化都是2。

Binfun的理解是,因为std::move的显式声明的关系,result &&s1 = 这种让右值生命值延长的方法失效了,最终s1指向的是已经被析构掉的右值地址(如上的0x7ffe7d59f380),而因为编译器优化等级的关系,编译器会重新回收并利用这个地址。所以在调用process(2)的时候会重新使用0x7ffe7d59f380这个地址。

如果编译选项的优化等级没那么高的话(以下把优化等级降为O1),会暂时避免这个问题:

g++ new_move.cpp -std=c++11 -O1 -fno-elide-constructors && ./a.out

打印如下:

---step1---
In process function
constructor(1) [0x7ffca4276e50]
moving from [0x7ffca4276e50] to [0x7ffca4276e80]
destructor() [0x7ffca4276e50]
destructor() [0x7ffca4276e80]
---step2---
In process function
constructor(2) [0x7ffca4276e50]
moving from [0x7ffca4276e50] to [0x7ffca4276e90]
destructor() [0x7ffca4276e50]
destructor() [0x7ffca4276e90]
---vals:---
s1 addr:[0x7ffca4276e80], val:[1]

可以看到以上0x7ffca4276e80虽然被析构了,但是没有那么快被重新利用起来,第二次使用的是0x7ffca4276e90,所以结果是正确的1。但是result &&s1 = std::move(process(1))这样的使用方式应该坚决不要用!

所以记住啊,这样可以:

result s1 = std::move(process(1)); //OK

这样也可以:

result &&s1 = process(1); //OK

这样不可以哦!:

result &&s1 = std::move(process(1)); //Error sometimes

参考链接

  1. https://www.cnblogs.com/tianfang/archive/2013/01/26/2878356.html
  2. https://zhuanlan.zhihu.com/p/107445960
  3. https://www.yhspy.com/2019/09/01/C-编译器优化之-RVO-与-NRVO/
  4. https://zhuanlan.zhihu.com/p/54050093
  5. http://stupefydeveloper.blogspot.com/2008/10/c-rvo-and-nrvo.html
  6. https://en.cppreference.com/w/cpp/language/reference
  7. https://zh.wikipedia.org/wiki/返回值优化
  8. https://en.wikipedia.org/wiki/Copy_elision
  9. https://zh.wikipedia.org/wiki/值_(電腦科學)
  10. 极客时间专栏《现代C++实战30讲》 03 | 右值和移动究竟解决了什么问题?

一段小代码秒懂C++右值引用和RVO(返回值优化)的误区的更多相关文章

  1. C#进阶系列——WebApi 接口返回值不困惑:返回值类型详解

    前言:已经有一个月没写点什么了,感觉心里空落落的.今天再来篇干货,想要学习Webapi的园友们速速动起来,跟着博主一起来学习吧.之前分享过一篇 C#进阶系列——WebApi接口传参不再困惑:传参详解  ...

  2. (转)C# WebApi 接口返回值不困惑:返回值类型详解

    原文地址:http://www.cnblogs.com/landeanfen/p/5501487.html 正文 前言:已经有一个月没写点什么了,感觉心里空落落的.今天再来篇干货,想要学习Webapi ...

  3. [转]C#进阶系列——WebApi 接口返回值不困惑:返回值类型详解

    本文转自:http://www.cnblogs.com/landeanfen/p/5501487.html 阅读目录 一.void无返回值 二.IHttpActionResult 1.Json(T c ...

  4. JavaScript 在函数中使用Ajax获取的值作为函数的返回值

    解决:JavaScript 在函数中使用Ajax获取的值作为函数的返回值,结果无法获取到返回值 原因:ajax默认使用异步方式,要将异步改为同步方式 案例:通过区域ID,获取该区域下所有的学校 var ...

  5. asp 获取url 返回值 和 对json 返回值的处理

    Function GetHttpPage(HttpUrl,endoce) If endoce = "" Then endoce = "GB2312" If Is ...

  6. 【Java 小实验】重写(覆写 Override)返回值类型能不能相同

    背景 每次看到重写那里写着: 重写机制是指子类的方法的方法名.参数表.返回值与父类中被重写的方法都相同,而方法体不同. 而重载是: 方法名与父类中的相同,而参数表不同,则属于同名方法的重载. 本来的感 ...

  7. C++引用和函数返回值

    这是老师上课讲的内容,现在把它写下来,一方面当做复习,另一方面真的想学点东西.废话不多说,先贴上测试的代码: #include <iostream.h> float temp; float ...

  8. WebApi接口返回值不困惑:返回值类型详解

    前言:已经有一个月没写点什么了,感觉心里空落落的.今天再来篇干货,想要学习Webapi的园友们速速动起来,跟着博主一起来学习吧.作为程序猿,我们都知道参数和返回值是编程领域不可分割的两大块,此前分享了 ...

  9. WebApi 接口返回值不困惑:返回值类型详解。IHttpActionResult、void、HttpResponseMessage、自定义类型

    首先声明,我还没有这么强大的功底,只是感觉博主写的很好,就做了一个复制,请别因为这个鄙视我,博主网址:http://www.cnblogs.com/landeanfen/p/5501487.html ...

随机推荐

  1. 7、Spring Boot检索

    1.ElasticSearch简介 Elasticsearch是一个分布式搜索服务,提供Restful API,底层基于Lucene,采用多shard(分片)的方式保证数据安全,并且提供自动resha ...

  2. 企业级工作流解决方案(七)--微服务Tcp消息传输模型之消息编解码

    Tcp消息传输主要参照surging来做的,做了部分裁剪和改动,详细参见:https://github.com/dotnetcore/surging Json-rpc没有定义消息如何传输,因此,Jso ...

  3. 掌握这些springboot的配置方式,让你工作效率翻个倍!

    springboot的多种配置方式 java配置主要靠java类和一些注解,比较常用的注解有: @Configuration :声明一个类作为配置类,代替xml文件 @Bean :声明在方法上,将方法 ...

  4. 用CorelDRAW来制作产品结构图的方法

    一.产品结构图的重要性 随着我国经济不断的高速发展,大家的生活水平不断提高,我们将会在生活生产中越来越多的,遇到许多各种各样的生产产品和生活消费品.科技的飞速进步,更是使这些产品.消费品包含了很强的科 ...

  5. LeetCode周赛#203 题解

    1561. 你可以获得的最大硬币数目 #贪心 题目链接 题意 有 3n 堆数目不一的硬币,你和你的朋友们打算按以下方式分硬币: 每一轮中,你将会选出 任意 3 堆硬币(不一定连续). Alice 将会 ...

  6. 第一次UML编程作业

    博客班级 https://edu.cnblogs.com/campus/fzzcxy/2018SE2/ 作业要求 https://edu.cnblogs.com/campus/fzzcxy/2018S ...

  7. 冰河教你一次性成功安装K8S集群(基于一主两从模式)

    写在前面 研究K8S有一段时间了,最开始学习K8S时,根据网上的教程安装K8S环境总是报错.所以,我就改变了学习策略,先不搞环境搭建了.先通过官网学习了K8S的整体架构,底层原理,又硬啃了一遍K8S源 ...

  8. MySQL重做日志(redo log)

    前面介绍了三种日志:error log.slow log.binlog,这三种都是 Server 层的.今天的 redo log 是 InnoDB引擎专有的日志文件. 为什么要有 redo log 用 ...

  9. 莫比乌斯反演进阶-洛谷P2257/HDU5663

    学了莫比乌斯反演之后对初阶问题没有任何问题了,除法分块也码到飞起,但是稍微变形我就跪了.用瞪眼观察法观察别人题解观察到主要内容除了柿子变形之外,主要就是对于miu函数的操作求前缀和.进而了解miu函数 ...

  10. 解决:com.netflix.discovery.shared.transport.TransportException: Cannot execute request on any known server

    com.netflix.discovery.shared.transport.TransportException: Cannot execute request on any known serve ...