什么是lvalue, 什么是rvalue?

lvalue: 具有存储性质的对象,即lvalue对象,是指要实际占用内存空间、有内存地址的那些实体对象,例如:变量(variables)、函数、函数指针等。

rvalue:相比较于lvalue就是所谓的没有存储性质的对象, 也就是临时对象。

也可以这样理解:

lvalue: 通过它能够找到内存中存放的变量(location value),位于赋值运算符左,可以赋值。
rvalue:存放在lvalue对应的内存中的东西(register value), 位于赋值运算符左,不可赋值。

对左值和右值的一个最常见的误解是:等号左边的就是左值,等号右边的就是右值。左值和右值都是针对表达式而言的,左值是指表达式结束后依然存在的持久对象,右值是指表达式结束时就不再存在的临时对象.

例如:

;
;
int *pFlag = &a;
vector<int> vctTemp;
vctTemp.push_back();
string str1 = "hello ";
string str2 = "world";
;  

请问,a,b, a+b, a++, ++a, pFlag, *pFlag, vctTemp[0], 100, string("hello"), str1, str1+str2, m分别是左值还是右值?、

  • a和b都是持久对象(可以对其取地址),是左值;
  • a+b是临时对象(不可以对其取地址),是右值;
  • a++是先取出持久对象a的一份拷贝,再使持久对象a的值加1,最后返回那份拷贝,而那份拷贝是临时对象(不可以对其取地址),故其是右值;
  • ++a则是使持久对象a的值加1,并返回那个持久对象a本身(可以对其取地址),故其是左值;
  • pFlag和*pFlag都是持久对象(可以对其取地址),是左值;
  • vctTemp[0]调用了重载的[]操作符,而[]操作符返回的是一个int &,为持久对象(可以对其取地址),是左值;
  • 100和string("hello")是临时对象(不可以对其取地址),是右值;
  • str1是持久对象(可以对其取地址),是左值;
  • str1+str2是调用了+操作符,而+操作符返回的是一个string(不可以对其取地址),故其为右值;
  • m是一个常量引用,引用到一个右值,但引用本身是一个持久对象(可以对其取地址),为左值。

再例如:

;

变量a的存储空间是 &a (a的lvalue), 存储空间&a上的值为5(a的rvalue).

再看一个例子:

int* p = NULL;
p = ); 

对于p:  lvalue为&p(p的地址), rvalue为&p地址上的值.

对于*p: lvalue为p(*p的地址),rvalue为p地址上的值.

有了这些,对于  左值引用就比较好理解了.

05-07 15:06