为了支持移动操作,C++11引入了一种新的引用类型,即右值引用。右值引用就是必须绑定到右值的引用,并通过&&获得。
右值引用可关联到右值,其中右值包括字面常量、表达式(例如x+y等)以及返回值的函数(前提:该函数返回的不是引用)。
#include<iostream>
using namespace std;
int main()
{
int x = 10;
int y = 23;
int&& r1 = 13;
int&& r2 = x + y;
cout << r1 << endl;//13
cout << r2 << endl;//33
r1 = 130;//修改r1的值
x = 17;//修改x的值
cout << r1 << endl;//130,右值引用的值可以修改
cout << r2 << endl;//33,没改变
return 0;
}
以上的右值引用的正确用法,注意:r2关联到的是当时计算x+y得到的结果。也就是说,r2关联到的是23,尽管修改了x的值,r2也不会改变。但是对于右值引用的字面常量,可直接像r1那样进行修改。
以下是左值引用和右值引用的相关用法:
#include<iostream>
using namespace std;
int main()
{
int x = 10;
int& r1 = x;//正确,r引用x
int& r2 = x * 20;//错误,x * 20是一个右值
const int& r3 = x * 20;//正确,可以将一个const的引用绑定到一个右值上
int&& rr1 = x;//错误,不能将一个右值引用绑定到一个左值上
int&& rr2 = x * 20;//正确,x * 20是一个右值,所以可以将rr2绑定到x * 20上
int&& rr3 = 30;//正确,字面常量是右值,可以将一个右值引用绑定到一个右值上
int&& rr4 = rr3;//错误,表达式rr1是左值,不能将一个右值引用绑定到一个左值上
return 0;
}
从上述代码中,我们可以看到
返回左值引用的函数,连同赋值、下标、解引用和前置递增/递减运算符,都是返回左值表达式的例子,因此我们可以将一个左值引用绑定到这类表达式的结果上;
而返回非引用类型的函数,连同算术、关系、位以及后置递增/递减运算符,都生成右值,因此我们不可以将一个左值引用绑定到这类表达式上,但是我们可以将一个const的左值引用或一个右值引用绑定到这类表达式上。
并且在最后,我们同样不能将一个右值引用绑定到一个右值引用类型的变量上,因为变量可以看作只有一个运算对象而没有运算符的表达式,而变量表达式都是左值,即使这个变量是右值引用类型也不行,故我们不能将一个右值引用绑定到一个左值(这里指右值引用类型的变量)上。
故左值引用和右值引用的区别:
左值持久,右值短暂;(左值有持久的状态,而右值要么是字面常量,要么是在表达式求值过程中创建的临时对象,且所引用的对象将要被销毁,该对象也没有其他用户)