左值和右值:
- 在C++中,左值可以出现在赋值语句的左边和右边;右值只能出现在赋值语句的右边,不能出现在赋值语句的左边。变量是左值,常量是右值。
- 对左值和右值的一个最常见的误解是:等号左边的就是左值,等号右边的就是右值。
- 左值和右值都是针对表达式而言的,左值是指表达式结束后依然存在的持久对象,右值是指表达式结束时就不再存在的临时对象。一个区分左值与右值的便捷方法是:看能不能对表达式取地址,如果能,则为左值,否则为右值。下面给出一些例子来进行说明
#include<iostream>
using namespace std;
void print(int& a)
{
cout << a;
}
int main()
{
print(10);// error C2664: “void print(int &)”: 无法将参数 1 从“int”转换为“int &”
return 0;
}
print(10)中的10是整形常量,是右值,不能对其进行修改,如果其传入print函数,print函数中的语句可能会对其进行修改,这就会导致错误。所以编译器不允许把右值传给非常量引用。即提示的错误“ 无法将参数 1 从“int”转换为“int &”,有时会提示成“非常量引用的初始值必须为左值”。可以把void print(int& a)改为void print(const int& a).这时变为了常量引用,就可以初始化为右值了。也可以把print(10)改为int t=10;print(t);这样print函数的实参就是左值了。
#include<iostream>
using namespace std;
void print(double& a)
{
cout << a;
}
int main()
{
int i = 10;
print(i);// “void print(double &)”: 无法将参数 1 从“int”转换为“double &”
return 0;
}
这里涉及到一个隐含转换,由于print函数的形参是double类型,而实参是int类型,所以会产生一个临时的中间变量,比如const double inner_temp=(double)i;临时的中间变量都是const,所以是右值,在把const double inner_temp传给double& a的时候自然会出错,可以把void print(double& a)改为void print(const double& a).