- 左右值的由来
左右值这两概念从c中传承而来,在c中左值是指能够出现在等号的左边的变量(表达式),右值是指只能出现在等号右边的变量(表达式)。 - 左右值的区别
左值的变量或表达式由两部分组成,分别为数据值以及内存地址,而右值没有内存地址。 - 简单的小例子
#include <iostream>
using namespace std;
void main(){
int i = 1;
int *ptr;
ptr = &(i++); // 这里是不正确的这里i++是右值
ptr = &(++i); // 正确,这里的++i是左值。
};
复制代码
- 为什么会这样
这里简单的解释一下i++
为什么返回的是右值而++i
返回的是左值。
// i++在编译器编辑执行时,内部流程如下:
int temp;
temp = i;
i = i + 1;
// 这里temp是个临时变量,它的值是i的数据值,所以temp是个常量,常量不能作为左值。
return temp;
// ++i 如下理解:
i = i + 1;
// 这里返回的是i本身。
return i;
复制代码
- 运算符重载
#include <iostream>
using namespace std;
class Test {
private:
int a;
int b;
public:
Test(void){
a = 0;
b = 0;
};
// ++i 运算符重载, 这里是左值所以返回一个地址
Test &operator++(){
++a;
if(a >= 2){
++b;
a -= 2;
};
// 返回当前对象,这里的this是指向当前对象的指针,如果想获取值需要通过*操作符获取。
return *this;
};
/*
这里i++需要传入参数,不要问我为什么,我也想知道c++11为啥这么规定。
个人理解是为了区分i++还是++i重载,像函数重载一样当编译器编译执行时,会自己匹配合适的重载函数。
*/
Test operator++(int){
// T拷贝当前对象
Test T = *this;
++a;
if(a >= 2){
++b;
a -= 2;
};
// 返回当前对象的拷贝类
return T;
};
void info(void){
std::cout<<"a: "<<a<<" b:"<<b<<std::endl;
};
};
int main(){
Test T1, T2;
++T1;
T1.info(); // a: 1 b:0
++T1;
T1.info(); // a: 0 b:1
T2++;
T2.info(); // a: 1 b:0
T2++;
T2.info(); // a: 0 b:1
system("pause");
return 0;
};
复制代码
ps: 博主还是新手村打B的小萌新,如果文章有误请指出,谢谢。