左值有右值的区别
每个表达式都会得到左值或右值。
左值(lvalue)是计算为一个持久存在的值,其内存地址可被用来持续存储值。右值(rvalue)被计算为一个暂时存储的结果。
之所以被称为lvalue,是因为lvalue的表达式通常出现在赋值运算符的左边(也可出现在运算符表达式的右边),而rvalue只能出现在赋值运算符的右边。
左值,可以由&操作符取出值的存储地址,而右值不行。
简单的分析
int a = 2;
int b = 3;
int c = (a + b);
cout << &a << &b << &c << endl; // 正确
a、b、c都是左值,上面三条语句是变量的定义,编译器会为a,b,c分配内存空间,三个变量的地址都是可知的。
(a+b)是右值,(a+b)的结果只是临时存储,在赋值给c后,就会舍弃(b+c结果的内存)。
cout << &(a+b) << endl; // error (a+b)是右值,取右值的地址是没有意义的。语句执行完毕后右值就会释放
// 地址上的值会发生改变,如果靠指针强行访问,是非常危险的行为,因此c++编译器不允许取右值的地址。
左右值在函数中的使用
#include <iostream>
using namespace std;
void fun(int a_b)
{
cout << &a_b << endl; // 正确
}
int main()
{
int a = 1, b = 3;
cout << &a << endl;
cout << &b << endl;
// cout << &(a+b) << endl; // error
fun(a+b);
return 0;
}
函数中定义的变量a_b,虽然只能在函数fun中访问,函数结束后被销毁,但a_b仍是左值。函数fun调用时,相当于执行int a_b = a+b;语句给参数a_b赋值。
点睛
看到这,你可能已经学会分辨左值和右值了,但你可能会存在疑惑:好像并没有左值和右值也没什么知识点。
没错,光知道这些还不够,要想真正的发挥右值的作用,我们还需要知道左值引用、右值引用、std::move和完美转发std::forward()。