1 概念
在C++或者C语言中,一个表达式(可以时字面量、变量、对象、函数的返回值等)根据其使用场景不同,分为左值表达式和右值表达式。确切的说C++中左值和右值的概念时从c语言继承过来的。
- 左值的英文简写为“lvalue”(locator value):意为存储在内存中,有明确存储地址(可寻址)的数据。
- 右值的英文简写为“rvalue”(read value):指的是那些可以提供数据值的数据(不一定可以寻址,例如存储于寄存器中的数据)。
判断某个表达式是左值还是右值的方法:
可位于赋值号(=)左侧的表达式就是左值;反之,只能位于赋值号右侧的表达式就是右值。
int a = 5;
5 = a; //错误,5 不能为左值
/*
其中a是一个左值,字面值5是一个右值
*/
int b = 10; // b 是一个左值
a = b; // a、b 都是左值,只不过将 b 可以当做右值使用
有名称的、可以获取到存储地址的表达式即为左值;反之则是右值。
上述示例中变量a、b是变量名且通过&a、&b可以获得他们得存储地址,因此a和b都是左值;反之,字面值5、10,它们既没有名称,也无法获取其存储地址(字面量通常存储在寄存器中,或者和代码存储在一起),因此5、10都是右值。
2 右值引用
在C++11之前就有引用,但是此种引用有一个缺陷,即正常情况下只能操作C++中得左值,无法对右值添加引用。
int num = 10;
int &a = num;
int &b = 10;//编译报错
但是允许使用个常量左值引用操作右值。也就是说,常量左值引用既可以操作左值,也可以操作右值。
int num = 10;
int &a = num;
const int &b = 10; //并没有实际用处.
std::cout << "&num:" << &num << endl;//&num:0x75bbffba8
std::cout << "&a:" << &a << endl;//&a:0x75bbffba8
std::cout << "&b:" << &b << endl;//&b:0x75bbffbac
为此,C++11标准新引入了另一种引用方式,称为右值引用,用“&&”表示。
int num = 10;
int &a = num;
int &&b = 10;
std::cout << "&num:" << &num << endl; //&num:0x75bbffba8
std::cout << "&a:" << &a << endl; //&a:0x75bbffba8
std::cout << "&b:" << &b << endl; //&b:0x75bbffbac
注意,引用b的地址并不是num的地址。
3 右值引用的应用场景
待补充