什么是lvalue, 什么是rvalue?
lvalue: 具有存储性质的对象,即lvalue对象,是指要实际占用内存空间、有内存地址的那些实体对象,例如:变量(variables)、函数、函数指针等。
rvalue:相比较于lvalue就是所谓的没有存储性质的对象, 也就是临时对象。
也可以这样理解:
lvalue: 通过它能够找到内存中存放的变量(location value),位于赋值运算符左,可以赋值。
rvalue:存放在lvalue对应的内存中的东西(register value), 位于赋值运算符左,不可赋值。
对左值和右值的一个最常见的误解是:等号左边的就是左值,等号右边的就是右值。左值和右值都是针对表达式而言的,左值是指表达式结束后依然存在的持久对象,右值是指表达式结束时就不再存在的临时对象.
例如:
int a = 10; int b = 20; int *pFlag = &a; vector<int> vctTemp; vctTemp.push_back(1); string str1 = "hello "; string str2 = "world"; const int &m = 1;
请问,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是一个常量引用,引用到一个右值,但引用本身是一个持久对象(可以对其取地址),为左值。
再例如:
int a = 5;
变量a的存储空间是 &a (a的lvalue), 存储空间&a上的值为5(a的rvalue).
再看一个例子:
int* p = NULL; p = new int(5);
对于p: lvalue为&p(p的地址), rvalue为&p地址上的值.
对于*p: lvalue为p(*p的地址),rvalue为p地址上的值.
有了这些,对于 左值引用就比较好理解了.