一、引子

我们所谓的左值、右值,正确的说法应该是左值表达式、右值表达式。

因为C++的表达式不是左值就是右值。

在C中,左值指的是既能够出现在等号左边也能出现在等号右边的表达式,右值指的则是只能出现在等号右边的表达式。

而在C++中,二者的区别就不是这么简单了。

二、关键点

【官方定义】

  1. 一个左值表达式的求值结果是一个对象或者一个函数,而某些右值表达式的求值结果也是对象;
  2. 以常量对象为代表的某些左值并不能作为赋值语句的左侧运算对象。
  • 归纳:当一个对象被用作右值的时候,用的是对象的值(内容);而被用作左值的时候,用的是对象的身份(在内存中的位置)。

【解读定义】

  解读第一句:既然左值和右值都可以是对象,那怎么区分一个对象是左值还是右值呢?

  • 左值:能用“取地址&”运算符获得对象的内存地址
  • 右值:不能用“取地址&”运算符获得对象的内存地址
  • 特例:因为可以用&取得字符串字面值常量的地址,所以它是一个左值;而其他的字面值常量则不可用&取地址,所以它们是右值

  解读第二句:字符串字面值常量确实不能放在等号左边啊,没错啊!

  解读归纳:所以啊,左值看地址,右值看内容。

【补充】

#include <iostream>

using namespace std;

int& func(int &a)
{
return a; // 不能返回局部对象的引用
} int main()
{
int a = 10;
func(a) = 100; // 返回非常量引用的函数可以当作左值使用
cout << a << endl; // 输出100
return 0;
}

三、为什么要区分左值右值

  • 不同的运算符对运算对象的要求各不相同,有的需要左值运算对象、有的需要右值运算对象;返回值也有差异,有的得到左值结果、有的得到右值结果。

  举例:使用关键字decltype的时候,其表达式的求值结果若是左值,decltype作用于该表达式(不能是变量)得到一个引用类型。例如,p的类型是int *,因为解引用运算符生成左值,所以decltype(*p)的结果是int &。而另一方面,因为取地址运算符生成右值,所以decltype(&p)的结果是int **,即结果是一个指向整型指针的指针。

05-04 03:14