简介

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

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

为什么要使用右值引用?

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

#include <iostream>
using namespace std; class Test
{
private:
public:
Test(/* args */) {
cout << "构造" << endl;
}
int a = 0;
~Test() {
cout << "析构" << endl;
} Test(const Test &a) {
cout << "复制" << endl;
}
}; Test GetTest() {
Test af;
return af;
} int main() {
Test a = GetTest();
return 0;
}

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

构造
复制
析构
复制
析构
析构

可以发现,为了保证正常运行和堆栈正常,程序对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()函数的代码改为

int main() {
Test &&a = GetTest();
return 0;
}

编译运行后得到额结果为

构造
复制
析构
析构

可以看到,程序减少了一次复制。这一次的减少就是上述的第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. element-ul二次封装table表格

    在项目中el的表格使用的地方太多了,若不进行封装,使用的时候页面会显得非常的冗余且难以维护,有时表格样式还不能做到一致:今天分享一个在工作中封装的表格 由于大多代码都在页面有介绍,就不在外面解释了 一 ...

  2. android开发之edittext弹出输入框遮挡住文字。解决方法

    在ManiFest清单文件中修改被遮挡的类的EditText android:windowSoftInputMode="adjustPan|stateHidden"

  3. 移动开发中如何整合HTML 5和原生代码

    移动开发中如何整合HTML 5和原生代码 https://blog.csdn.net/lvjin110/article/details/41038565

  4. Salesforce LWC学习(二十三) Lightning Message Service 浅谈

    本篇参考: https://trailhead.salesforce.com/content/learn/superbadges/superbadge_lwc_specialist https://d ...

  5. 汽车芯片如何高效符合ISO 26262功能安全标准

      汽车芯片和集成电路(IC)是高级驾驶员辅助系统(advanced driver assistance systems-ADAS)和联网自动驾驶汽车(connected autonomous veh ...

  6. vue安装pubsub-js 库的命令

    1.查看pubsub-js 库是否已经存在该库命令: npm info pubsub-js 2.若不存在,则先安装pubsub-js 库,命令如下: npm install --save pubsub ...

  7. vue-devtools-4.1.4_0.crx及Vue.js not detected的问题

    谷歌-更多工具-扩展程序 Vue.js not detected的问题

  8. get_started_3dsctf_2016

    题外:这道题不是很难,但是却难住了我很久.主要是在IDA中查看反编译出的伪代码时双击了一下gets()函数,结果进入gets函数内部,我当时就懵了,误以为这是一个自定义函数,但是自定义函数应该应该不能 ...

  9. Android 重构方案

    前言 最近面试了很多候选人,发现很多同学在简历上都写得非常厉害,负责架构设计,项目重构之类的.但是问起来,很多人都说不出个所以然来.今天我们不谈架构设计,我们聊一下重构.我面试时候经常会问,你是怎么重 ...

  10. django之admin配置

    要在admin内显示的表,在admin中进行注册,然后在登录admin后台,才可以对表进行操作例如:from django.contrib import adminfrom app01 import ...