简介

关于C++中的右值引用的详细可以看这一批博文《从4行代码看右值引用》。那一篇博文详细结合四行简单的代码详细介绍了右值引用的使用方法和一些场景,非常实用。

而本篇博文主要介绍一下我在学习右值引用的一些心得。因为在学习右值引用的时候,有一些地方非常难理解。所以写下这一篇博文,防止遗忘,由于对于C++涉猎不多,所以有一些不正确的地方,欢迎斧正。

为什么要使用右值引用?

对于普通开发者来说,右值引用的好处在于可以将保证在取用函数返回值的时候,减少一次复制的机会。如:

  1. #include <iostream>
  2. using namespace std;
  3. class Test
  4. {
  5. private:
  6. public:
  7. Test(/* args */) {
  8. cout << "构造" << endl;
  9. }
  10. int a = 0;
  11. ~Test() {
  12. cout << "析构" << endl;
  13. }
  14. Test(const Test &a) {
  15. cout << "复制" << endl;
  16. }
  17. };
  18. Test GetTest() {
  19. Test af;
  20. return af;
  21. }
  22. int main() {
  23. Test a = GetTest();
  24. return 0;
  25. }

使用编译选项-fno-elide-constructors,可以得到结果

  1. 构造
  2. 复制
  3. 析构
  4. 复制
  5. 析构
  6. 析构

可以发现,为了保证正常运行和堆栈正常,程序对Test对象经过了一次构造和两次复制。分别是

  1. GetTest()函数中Test af的构造,由于af是GetTest()的临时变量,因此此时使用的内存空间是GetTest()里面的栈1
  2. 为了把af从GetTest()的栈中取出来,于是将把af指向的Test对象复制到了main()的栈,由于程序“并不知道”接下来的操作,所以此时对象的存在就是一个临时的。语句执行完,它就析构了。
  3. 为了保存这一个对象,我们使用了a,而为了将上一步得到的值复制到a,程序又进行了一次复制。

以上便是程序的全过程。而可以很清楚的发现,第2次复制得到的值和第3次复制得到的值,它们都位于main()的堆栈空间内的,他们的生命周期都是一样的,和main()一样长。那么我们有没有办法可以减少第3次的复制呢?

在没有右值引用的时候有两种方法。

  1. 靠编译器优化2
  2. 使用const Test&,常量引用。

第一种方法不表,以目前的编译器优化,的确可以获得非常多的优惠,但是这些编译器提供的,并不是语法提供的,不同的编译器可能存在不一样的表现。第二种在需要改变返回值的时候非常麻烦。

而右值引用可以解决这一问题。我们将main()函数的代码改为

  1. int main() {
  2. Test &&a = GetTest();
  3. return 0;
  4. }

编译运行后得到额结果为

  1. 构造
  2. 复制
  3. 析构
  4. 析构

可以看到,程序减少了一次复制。这一次的减少就是上述的第3次的复制。而且此时的变量a可以随意的更改。

那我们可以把第二次的复制也去掉吗?当然可以,比如把GetTest()的返回值改为Test&&,然后给af加上std::move强制转换成右值。但是,这样的操作破坏了程序的堆栈,会出现很大的问题。

右值引用和unique_ptr

unique_ptr是C++中智能指针的一部分,另外还有auto_ptrshared_ptr。但是刚刚使用C++不久,所以只对shared_ptr有一定的了解,而auto_ptr由于一些原因已经被弃用了。unique_ptr只允许一个智能指针对象拥有指针。而shared_ptr则允许多个,其内部有一个引用计数器,当引用为0时便释放空间,因此相对于unique_ptr,其所占的内存空间较大,使用的时候,效率也相对较低。

unique_ptr的实现必须保证不共享其指针,不复制到其他的unique_ptr,也不能使用值传递到函数里面。要将其所有权交给其他变量,只能通过“移动”的操作。

为了保证移动的语义上的正确性,被移动的unique_ptr必须为右值。因为右值是被认为是即将消亡的值(将亡值),所以移动操作可以把被移动的unique_ptr的引用值为nullptr。

std::move和std::forward

std::move是将一个左值转换为右值,在内部实现直接使用了强制转换static_cast<T&&>()。它一般用于将左值转换为右值,在unique_ptr转移的时候需要使用。一般来说,使用了move后,该对象就不应该继续使用了。

std::forward即完美转发,保证参数在传递过程中属性不变,右值还是右值,左值还是左值。

附录

  1. 此处表达可能有一些不正确,因为所有的函数调用及其局部变量应该都是在一个栈上。但是在函数结束的时候,必须要恢复栈的指针。这一步便需要把其局部变量全部删除,因此可以认为其局部变量是位于函数的栈中。
  2. 去掉编译选项-fno-elide-constructors

    ------------恢复内容结束------------

关于C++的右值引用的一些看法的更多相关文章

  1. C++右值引用浅析

    一直想试着把自己理解和学习到的右值引用相关的技术细节整理并分享出来,希望能够对感兴趣的朋友提供帮助. 右值引用是C++11标准中新增的一个特性.右值引用允许程序员可以忽略逻辑上不需要的拷贝:而且还可以 ...

  2. C++ 11 中的右值引用

    C++ 11 中的右值引用 右值引用的功能 首先,我并不介绍什么是右值引用,而是以一个例子里来介绍一下右值引用的功能: #include <iostream>    #include &l ...

  3. 图说函数模板右值引用参数(T&&)类型推导规则(C++11)

    见下图: 规律总结: 只要我们传递一个基本类型是A④的左值,那么,传递后,T的类型就是A&,形参在函数体中的类型就是A&. 只要我们传递一个基本类型是A的右值,那么,传递后,T的类型就 ...

  4. c++11的右值引用、移动语义

    对于c++11来说移动语义是一个重要的概念,一直以来我对这个概念都似懂非懂.最近翻翻资料感觉突然开窍,因此记下.其实搞懂之后就会发现这个概念很简单,并无什么高深的地方. 先说说右值引用.右值一般指的是 ...

  5. VS2012 error C2664: “std::make_pair”:无法将左值绑定到右值引用

    在vs2012(c++)make_pair()改动: C++: template <class T1, class T2> pair<V1, V2> make_pair(T1& ...

  6. 右值引用、move与move constructor

    http://blog.chinaunix.net/uid-20726254-id-3486721.htm 这个绝对是新增的top特性,篇幅非常多.看着就有点费劲,总结更费劲. 原来的标准当中,参数与 ...

  7. 【转】C++11 标准新特性: 右值引用与转移语义

    VS2013出来了,对于C++来说,最大的改变莫过于对于C++11新特性的支持,在网上搜了一下C++11的介绍,发现这篇文章非常不错,分享给大家同时自己作为存档. 原文地址:http://www.ib ...

  8. move语义和右值引用

    C++11支持move语义,用以避免非必要拷贝和临时对象. 具体内容见收藏中的“C++右值引用” .

  9. [转载] C++11中的右值引用

    C++11中的右值引用 May 18, 2015 移动构造函数 C++98中的左值和右值 C++11右值引用和移动语义 强制移动语义std::move() 右值引用和右值的关系 完美转发 引用折叠推导 ...

随机推荐

  1. 5分钟快速学会xpath定位

    今天我们先来和大家说一下appium,首先教大家如何定位xpath,五分钟即可学会:例:现在我想定位下面这个登录按钮: xpath该怎么写呢? 先不管三七二十几,先写//,然后找你要定位元素最近的可以 ...

  2. 服务发现Eureka、zookeeper、consul

    Spring Cloud为开发人员提供了工具,以快速构建分布式系统中的某些常见模式(例如,配置管理,服务发现,断路器,智能路由,微代理,控制总线,一次性令牌,全局锁,领导选举,分布式会话,群集状态). ...

  3. PJSIP开发指南

    一.通用设计 1.1   架构 1.1.1        通信图 下面的图展示了SIP消息在PJSIP组件间从后端到前端如何传递的. 1.1.2        类图 下面的图显示类视图 1.2   E ...

  4. Android开发之将Edittext输入弹出的软键盘设置搜索确定键。详细代码,已验证可用。

    1,首先在布局中添加 android:imeOptions="actionSearch 2,然后java代码中设置代码: et_input_password.setOnEditorActio ...

  5. oeasy教您玩转linux010103我在那pwd

    我在哪pwd 回忆上次内容 我们上次列出了关于发行版详细信息文件的位置 ls /etc/*release 其中 ls是命令的名字,意味着list列表 /是整个文件系统的根root /etc是根目录下的 ...

  6. 0 mysql 安装

    1 安装网址 https://dev.mysql.com/downloads/installer/ 选择 mysql server版本一路next 2.配置环境 mysql 默认安装位置是: C:\P ...

  7. 【CF】Sereja and Arcs

    #include <bits/stdc++.h> #define llong long long using namespace std; const int N = 1e5; const ...

  8. H5简单内容

    1.简单认识H5 HTML5不仅仅是作为HTML标记语言的一个最新版本,更重要的是它指定了Web开发的一系列标准,成为第一个将Web作为应用开发平台的HTML语言. 我们日常讨论的H5其实是有一个泛称 ...

  9. SQL Node 1.05版

    输出: select a.f1, b.f2 from table01 a, ( select a from tb ) b where a.f1=1 and b.f2=2 or b.f3=3 order ...

  10. Brup sute

    暴力破解