天天看点

C++中左值引用和右值引用的区别

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右值

参考文档

继续阅读