01 什么是左值和右值?
- 左值: 就是有确定的内存地址、有名字的变量,可以被赋值,可以在多条语句中使用;
- 右值: 没有名字的临时变量,不能被赋值,只能在一条语句中出现,如:字面常量和临时变量。
02 如何区分左值与右值?
看能不能对表达式取地址
能否用“取地址&”运算符获得对象的内存地址。
对于临时对象,它可以存储于寄存器中,所以是没办法用“取地址&”运算符;
对于常量,它可能被编码到机器指令的“立即数”中,所以是没办法用“取地址&”运算符;
这也是C/C++标准的规定。
返回左值的表达式:返回左值引用的函数、赋值、下标、解引用、前置递增/递减运算符
。
返回右值的表达式:返回非引用类型的函数、算术、关系、位、后置递增/递减运算符
。
举例:
int a = 5;
int b = 6;
int *ptr = &a;
vector v1;
string str1 = "hello " ;
string str2 = "world" ;
const int &m = 1 ;
a+b//临时对象
a++//右值,是先取出持久对象a的一份拷贝,再使持久对象a的值加1,最后返回那份拷贝,而那份拷贝是临时对象(不可以对其取地址),故其是右值。
++a//左值,使持久对象a的值加1,并返回那个持久对象a本身(可以对其取地址),故其是左值。
v1[0]//调用了重载的[]操作符,而[]操作符返回的是一个int &,为持久对象(可以对其取地址)是左值。
string("hello")//临时对象(不可以对其取地址),是右值;
str1+str2//调用了+操作符,而+操作符返回的是一个string(不可以对其取地址),故其为右值;
str1//左值
*p//左值
03 左值引用和右值引用
C++11引入右值引用,c++11中用&表示左值引用,用&&表示右值引用。
int i = 2;//左值
int &a = i; //左值引用
int &&a = 10; //右值引用
根据修饰符的不同,左值引用可分为:非const左值、const左值;右值引用可分为:非const右值、const右值。
int i = 10;//非const左值
const int j = 20;
//非const左值引用
int &a = i;//非const左值引用 绑定到 非const左值,编译通过
int &a = j;//非const左值引用 绑定到 const左值,编译失败
int &a = 5;//非const左值引用 绑定到 右值,编译失败
//const左值引用
const int &b = i;//const左值引用 绑定到 非const左值,编译通过
const int &b = j;//const左值引用 绑定到 const左值,编译通过
const int &b = 5;//const左值引用 绑定到 右值,编译通过
//非const右值引用
int &&c = 30;//非const右值引用 绑定到 右值,编译通过
int &&c = i;//非const右值引用 绑定到 非const左值,编译失败(不能将一个右值引用绑定到左值上)
//const右值引用
const int &&d = 40;//const右值引用 绑定到 右值,编译通过
const int &&d = c;//const右值引用 绑定到 非常量右值,编译通过
- 非常量的左值引用只能被绑定到非常量左值
- 非常量的右值引用只能被绑定到非常量右值
04 总结
- 非常量左值引用只能绑定到
非常量左值
,不能绑定到常量左值、非常量右值和常量右值; - 常量左值引用可以绑定到
const左值、非const左值、const右值、非const右值
。 - 非const右值引用只能绑定到
非const右值
; - const右值引用可绑定到
const右值和非const右值
。