我们在定义变量或函数传参的时候编译器经常会提示“表达式必须是可以修改的左值”,“非常量引用的初始值必须是左值”等等报错,本文将对左值和右值,左值引用和右值引用做一个简单的总结。
1. 左值和右值
大多数情况下来说左值是=左边的值,右值是=右边的值,如
int a = 10
此时a为左值, 10为右值;
但是对于
int a = 10;
int b = a;
尽管a在=的右侧 ,但a仍为左值。
可以简要概括为在内存中有位置的变量为左值,而对于没有储存空间、没有位置的临时变量则为右值;我们可以给有储存位置左值赋值,但不能给没有储存位置的右值复制。
int a = 10;
20 = a; error:表达式必须是可以修改的左值
2. 左值引用和右值引用
很多情况下,为了节省内存,提高效率,函数参数值通常为引用,当我们不希望传递的引用被修改时可以加上const如
void func(const int &a){}
2.1 对于无const的引用
#include <iostream>
void func(int& a) {
std::cout << a << std::endl;
}int main() {
{
int a = 10;
func(a);
func(10); error: 非常量引用的初始值必须为左值
}
}
可以看到现在我们只能给函数传递左值,即左值引用。
2.2 对于带const的情况
#include <iostream>
void func(const int& a) {
std::cout << a << std::endl;
}int main() {
{
int a = 10;
func(a);
func(10); //right
}
}
当传递的参数为右值时,编译器会帮助我们生成一个临时变量,该情况下可以传递左值和右值。
2.3 右值引用
#include <iostream>
void func(int&& a) {
std::cout << a << std::endl;
}int main() {
{
int a = 10;
func(a); error: 无法将右值引用绑定到左值
func(10);
}
}
此时,我们只能传递右值(临时变量),即右值引用。
右值引用的一个好处是传递的是临时变量,它的作用和存在时间有限,在上面的例子中他只存在于func(10)的作用域内,因此你可以随意的操作它;但对于左值引用,它可能被多个函数使用,因此不能随意操作。