【C++】 左值和右值

C++----到底啥是左值和右值


<<C++primer>>几乎全篇都在讲左值和右值,那到底什么是左值和右值?
左值和右值源于c语言,原本为了记忆:左值可以位于赋值语句的左侧,右值则不能。

那什么是c++中的左值和右值呢?

总的来说,C++把左值看作是一个关联了名称的内存位置 ,相对而言,右值则是一个临时值,不能被程序的其他部分去访问。

c++11对C++98定义的右值进行了扩充,分为纯右值和将亡值,其中,纯右值等同于c++98中的右值概念,指的是临时变量和不与对象关联的字面量值,而将亡值则是跟右值引用有关的表达式,表达式通常是将要被移动的对象(移为他用),比如返回右值引用T&&的函数返回值、std::move的返回值,或者转换为T&&的类型转换函数的返回值。将亡值可以理解为通过“盗取”其他变量内存空间的方式获取到的值

字符串字面量是左值,它被定义为char N,const char N,可以去地址

这是什么意思,懵逼!

give you code!

// 定义一个函数 返回两数之和
int sub(int a,int b){
	return a+b;
}

int main(){
	int x = 1;
	int y = 2;
	x = 10;
	int z = sub(x,y);
	return 0;
}

从上面代码可以看出,x和y在定义初始化的时候,编辑器会在把内存中开辟一个空间存储,因此他们可以被其他部分访问到,比如可以作为函数的参数被调用,因此它们是左值。

而函数表达式**sub(x,y)**去是一个右值,因为它仅仅是一个临时内存位置,保存函数返回值,在程序离开函数作用域之后就被释放掉了,再也不能别访问了。

那可以去重复访问这个值吗?事实上是可以的
C++11引入了右值引用的概念,在讲右值引用之前,我们先看想一下左值引用(一般将的都是左值引用)

int a =3;
int &lRef = a;

从代码可以看出,a是一个左值,它有特定的内存位置,lRef 利用&去指向a,相当于给a取了个别名,事实上,a就是lRef ,lRef 就是a。
修改lRef 的话,a也会被修改。

左值引用和右值引用

  • 左值引用:传统的C++中引用被称为左值引用

  • 右值引用:C++11中增加了右值引用,右值引用关联到右值时,右值被存储到特定位置,右值引用指向该特定位置,也就是说,右值虽然无法获取地址,但是右值引用是可以获取地址的,该地址表示临时对象的存储位置

常引用

其中,常左值引用是一个特殊的存在,它可以关联左值,也可以关联右值

  1. 对某个变量(或表达式)建立常引用时,允许发生类型转换,而一般的引用则不允许
#include 
#include 
using namespace std;

int Max(const int& a, const int& b)
{
    return (a>b)?a:b;
}

int main(int argc,char* argv[])
{
    char c='a';
    const int &rc=c;  // 去掉const就不行  常引用可以实现类型转换
    cout<<(void*)&c<<endl;
    cout<<(void*)&rc<<endl;

    int i=7;
    const int &ri=i;
    cout<<(void*)&i<<endl;
    cout<<(void*)&ri<<endl;
    cout<<Max(rc,5.5)<<endl;
    getchar();
}

对一个表达式建立常引用时,如果该表达式的结果可以寻址,并且表达式的数据类型与引用类型相同,那么可以直接将该表达式结果的地址送入引用变量。此例中,&i和&ri的值相等就说明了这一点。否则,若表达式的数据类型与引用类型不相同,或是表达式结果不可寻址,那么只能另外建立一个无名临时变量存放表达式的结果(或其转换后的值),然后将引用于无名临时变量绑定

理解?
常左值引用关联右值时,会先用一个临时的右值引用绑定这个右值,然后再将引用与这个右值引用(临时变量)绑定?

也就是说,能够发生类型转换是常左值引用“乱杀”的一个大杀器


那右值引用又是啥?
对于上面的函数,我们添加几条语句

int && rRef = sub(a,b); 

不同于左值引用的一个&,右值引用需要用两个&&,给这个临时内存分配名称的话,程序的其他部分就可以访问。

需要注意,右值引用不能用在左值上!

有意思的是,在右值引用初始化完成之后,编辑器会给rRef分配内存, 它变成了左值!

屠龙的少年终将变成巨龙!

右值引用的特点

  1. 通过右值引用的声明,右值又“重获新生”,其生命周期与右值引用类型变量的生命周期一样长,只要该变量还活着,该右值临时量将会一直存活下去(夺舍重生)
  2. 右值引用独立于左值和右值。意思是右值引用类型的变量可能是左值也可能是右值(雌雄难辨)
  3. T&& t在发生自动类型推断的时候,它是左值还是右值取决于它的初始化(完美转发)

加油!
梦想不会太远!

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值