C++----到底啥是左值和右值
<<C++primer>>几乎全篇都在讲左值和右值,那到底什么是左值和右值?
左值和右值源于c语言,原本为了记忆:左值可以位于赋值语句的左侧,右值则不能。
那什么是c++中的左值和右值呢?
总的来说,C++把左值看作是一个关联了名称的内存位置 ,相对而言,右值则是一个临时值,不能被程序的其他部分去访问。
c++11对C++98定义的右值进行了扩充,分为纯右值和将亡值,其中,纯右值等同于c++98中的右值概念,指的是临时变量和不与对象关联的字面量值,而将亡值则是跟右值引用有关的表达式,表达式通常是将要被移动的对象(移为他用),比如返回右值引用T&&的函数返回值、std::move的返回值,或者转换为T&&的类型转换函数的返回值。将亡值可以理解为通过“盗取”其他变量内存空间的方式获取到的值
这是什么意思,懵逼!
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中增加了右值引用,右值引用关联到右值时,右值被存储到特定位置,右值引用指向该特定位置,也就是说,右值虽然无法获取地址,但是右值引用是可以获取地址的,该地址表示临时对象的存储位置
常引用
其中,常左值引用是一个特殊的存在,它可以关联左值,也可以关联右值
- 对某个变量(或表达式)建立常引用时,允许发生类型转换,而一般的引用则不允许
#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分配内存, 它变成了左值!
屠龙的少年终将变成巨龙!
右值引用的特点
- 通过右值引用的声明,右值又“重获新生”,其生命周期与右值引用类型变量的生命周期一样长,只要该变量还活着,该右值临时量将会一直存活下去(夺舍重生)
- 右值引用独立于左值和右值。意思是右值引用类型的变量可能是左值也可能是右值(雌雄难辨)
- T&& t在发生自动类型推断的时候,它是左值还是右值取决于它的初始化(完美转发)
加油!
梦想不会太远!