C++11中的右值引用和move函数

新版的C++标准库出现了很多C++11的新特性,刚开始接触确实很费劲,特别是右值引用和move函数这种基于效率的考虑损失了语言的简单直接的特性,废话不多说,先看右值引用。

C++中根据const和non_const,lvalue和rvalue可分为四类对象



   non_const          const

lvalue  非常量左值         常量左值

rvalue  非常量右值         常量左值



如果使用C++03标准,函数的参数选择const reference(常量引用),可以接受所有的传值类型:常量/非常量的左值/右值,即:

void f(const T& t);

可以接受所有T类型的对象传入。

而void f(T& t);

只能接受非常量的左值传入。

我们没有办法在函数参数类型区分只接受左值,或右值,要办到这件事,需要引入右值引用的概念。

void f(T&& t);

T&&不是引用的引用,是右值引用。C++11(准确的说是C++09)引入此项特性的目的是了健全左右值的引用。

为什么要区分?

考虑下面的函数,它已经被用烂了:

vector<int> f(vector<int>& p)
{
    vector<int> a;
    for(int i = 0 ;i < 100; i++)
        a.push_back(p+i);
    return a;
}vector<int> test = f(v);//v is a vector

这样的函数行为导致vector<int>a的初始化是一种浪费的行为,它创建了一段在堆上的空间存放数据,再把这些数据拷贝给vector<int> test。

我们完全可以把vector<int> a在堆上申请的空间以及被设置好的数据直接给test,告诉test你拿去用好了,反正我马上就死了。

好了,如果我们的vector中有一个构造函数,能够拿test的空间和数据,那么上面的过程也就完成了:

vector<int>(vector<int>&& v)//vector<int>的构造函数的重载版本
{//这是简化版的vector,只是为了让你看懂move的意义所在
    ptr = v.ptr;//假设ptr是指向存储地址的指针
    size = v.size;//size是vector中存储空间的大小
   ....  v.ptr = 0;//neccessary!
}

注意到构造函数的参数是vector<int>&&,它只接受右值引用的传入。

为什么要只接受右值的引用?因为,我们只想要马上就销毁的对象才把数据move过来,而其他还要继续使用的对象,数据是拷贝过来的(通过其他的拷贝构造函数的重载版本实现)

同时,我们的f函数需要修改如下:

vector<int> f(vector<int>& p)//注意返回类型不是vector<int>&&
{
    vector<int> a;
    for(int i = 0 ;i < 100; i++)
        a.push_back(p+i);
    return std::move(a);
}

move的作用是把对象修改为一个右值的引用

好了,如果没有右值引用,我们就没法完成上述功能。

推荐一篇博文,讲得比较详细http://blog.csdn.net/pongba/article/details/1684519

时间: 2024-11-20 11:40:09

C++11中的右值引用和move函数的相关文章

C++11中的右值引用及move语义编程

C++0x中加入了右值引用,和move函数.右值引用出现之前我们只能用const引用来关联临时对象(右值)(造孽的VS可以用非const引用关联临时对象,请忽略VS),所以我们不能修临时对象的内容,右值引用的出现就让我们可以取得临时对象的控制权,终于可以修改临时对象了!而且书上说配合move函数,可以大大提高现有C++的效率.那么是怎样提高它的效率的呢?看段代码先! #include <iostream> #include <utility> #include <vector

C++ 11 中的右值引用

C++ 11 中的右值引用 右值引用的功能 首先,我并不介绍什么是右值引用,而是以一个例子里来介绍一下右值引用的功能: #include <iostream>    #include <vector>    using namespace std; class obj    {    public :        obj() { cout << ">> create obj " << endl; }        obj(c

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

C++11中的右值引用 May 18, 2015 移动构造函数 C++98中的左值和右值 C++11右值引用和移动语义 强制移动语义std::move() 右值引用和右值的关系 完美转发 引用折叠推导规则 特殊模板参数推导规则 解决完美转发问题 引用 在C++98中有左值和右值的概念,不过这两个概念对于很多程序员并不关心,因为不知道这两个概念照样可以写出好程序.在C++11中对右值的概念进行了增强,我个人理解这部分内容是C++11引入的特性中最难以理解的了.该特性的引入至少可以解决C++98中的

C++11中的右值引用

原文出处:http://kuring.me/post/cpp11_right_reference May 18, 2015 移动构造函数 C++98中的左值和右值 C++11右值引用和移动语义 强制移动语义std::move() 右值引用和右值的关系 完美转发 引用折叠推导规则 特殊模板参数推导规则 解决完美转发问题 引用 在C++98中有左值和右值的概念,不过这两个概念对于很多程序员并不关心,因为不知道这两个概念照样可以写出好程序.在C++11中对右值的概念进行了增强,我个人理解这部分内容是C

【转载】C++ 11中的右值引用

本篇随笔为转载,原博地址如下:http://www.cnblogs.com/TianFang/archive/2013/01/26/2878356.html 右值引用的功能 首先,我并不介绍什么是右值引用,而是以一个例子里来介绍一下右值引用的功能: #include <iostream>    #include <vector>    using namespace std; class obj    {    public :        obj() { cout <&l

[转载]如何在C++03中模拟C++11的右值引用std::move特性

本文摘自: http://adamcavendish.is-programmer.com/posts/38190.htm 引言 众所周知,C++11 的新特性中有一个非常重要的特性,那就是 rvalue reference,右值引用. 引入它的一个非常重要的原因是因为在 C++ 中,常常右值,通俗地讲"在等号右边的"临时变量或者临时对象,我们是无法得到它的修改权限的. 由于类的构造和析构机制,往往产生的临时变量或临时对象的拷贝构造及析构,会带来不少的时间.资源消耗. 也同样由于这样的限

C++11标准之右值引用(rvalue reference)

1.右值引用引入的背景 临时对象的产生和拷贝所带来的效率折损,一直是C++所为人诟病的问题.但是C++标准允许编译器对于临时对象的产生具有完全的自由度,从而发展出了Copy Elision.RVO(包括NRVO)等编译器优化技术,它们可以防止某些情况下临时对象产生和拷贝.下面简单地介绍一下Copy Elision.RVO,对此不感兴趣的可以直接跳过: (1) Copy Elision Copy Elision技术是为了防止某些不必要的临时对象产生和拷贝,例如: struct A { A(int)

深入右值引用,move语义和完美转发

深入右值引用,move语义和完美转发 转载请注明:http://blog.csdn.net/booirror/article/details/45057689 乍看起来,move语义使得你可以用廉价的move赋值替代昂贵的copy赋值,完美转发使得你可以将传来的任意参数转发给 其他函数,而右值引用使得move语义和完美转发成为可能.然而,慢慢地你发现这不那么简单,你发现std::move并没有move任何东西,完美转发也并不完美,而T&&也不一定就是右值引用-- move语义 最原始的左值

左值、右值、右值引用与move()、forward()

1.左值(lvalue):可以进行取地址(&)运算的是左值.或者有时候可以理解为 既能够出现在等号左边也能出现在等号右边的变量(或表达式). 2.右值(rvalue):不可以进行取地址(&)运算的是右值.或者有时候可以理解为 只能出现在等号右边的变量(或表达式). 常见的右值有 字面量.函数返回的临时对象,匿名对象等. 以上判断一个对象是左值还是右值并不完全正确. const int c_a = 10; //左值,但是不能被赋值,也就不能出现在 = 左边 //字符串字面值,可以取地址,是左