简介

关于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. 深入理解C++中的new/delete和malloc/free动态内存管理

    malloc/free和new/delete的区别 malloc/free是C/C++标准库的函数:new/delete是C++操作符. malloc/free只是动态分配内存空间/释放空间:new/ ...

  2. 【Android】Android开发启动app弹出一张广告图片,Dialog可以查看大图,查看某个图片功能

    作者:程序员小冰,GitHub主页:https://github.com/QQ986945193 新浪微博:http://weibo.com/mcxiaobing 首先给大家看一下我们今天这个最终实现 ...

  3. 在Mac上打开多个Unity实例

    alias koa_unity="open -n /Applications/Unity\ 5.2.2/Unity.app" alias rob_unity="open ...

  4. UI设计中的软件知识

    最近挺想学学UI的,因为我们公司没有UI,所以做页面都是全靠摸索,老是被领导说没有审美[捂脸] 学习UI所需要的软件 PS  AI Sketch XD Sketch是MAC才能安装的软件 作者:彼岸舞 ...

  5. Java得到指定日期的时间

    //得到指定日期(几天前/几天后)整数往后推,负数往前移动private Date getAppointDay(int num) throws ParseException { DateFormat ...

  6. TP6.0 获取请求对象的五种方式

    目录 1. 门面类 2. 依赖注入 3. 框架提供的基础控制器的 request 属性 4. request() 助手函数 5. app() 超级助手函数 think\Request.think\fa ...

  7. [LeetCode]66. 加一(数组)

    ###题目 给定一个由整数组成的非空数组所表示的非负整数,在该数的基础上加一. 最高位数字存放在数组的首位, 数组中每个元素只存储单个数字. 你可以假设除了整数 0 之外,这个整数不会以零开头. 示例 ...

  8. Fiddler无法抓取web项目中的http请求解决方案

    问题:webform项目中对接API使用Fiddler无法获取该API的请求,该webform比较老的一个项目, 同一个API写在控制台可以抓取到请求,用web项目放在本地IIS却不行,使用IIS E ...

  9. Linux实战(15):Centos 防火墙常见端口命令

    开启命令 firewall-cmd --zone=public --add-port=80/tcp --permanent 关闭命令 firewall-cmd --zone=public --remo ...

  10. docker注册&打包

    在https://hub.docker.com上注册账号,新建reosity docker images命令列出dockerlist docker login命令登录:这其中出现了一个问题,登录不上去 ...