在C++中,左值(lvalue)和右值(rvalue)是表达式分类的关键概念,它们主要影响表达式的赋值、函数调用以及操作符的使用方式。这些概念在C++11及以后的版本中变得更加重要,因为引入了移动语义和右值引用,以优化资源管理和提高性能。
左值(Lvalue)
左值是指那些可以出现在赋值表达式左边的表达式。左值表示一个持久的对象,它占用内存中的某个位置,并且具有持久的身份(即地址)。左值可以是变量、函数返回左值引用的表达式等。左值可以被多次赋值,因为它们代表内存中的具体位置。
右值(Rvalue)
右值是指那些只能出现在赋值表达式右边的表达式。在C++11之前,右值主要指的是临时对象(temporary objects),即那些没有名称的对象,比如函数返回的临时对象、字面量等。右值通常表示一个短暂的值,它们不占用持久的内存位置,或者即使占用,也不允许我们对其进行后续操作(比如赋值)。
C++11及以后的变化
在C++11中,引入了右值引用的概念(使用&&表示),以及与之相关的移动语义和完美转发。这使得右值的处理变得更加灵活和高效。
右值引用:允许我们绑定到右值上,从而可以对临时对象或字面量进行操作,而不会导致额外的复制或移动。
移动语义:通过允许资源从一个对象“移动”到另一个对象(而不是复制),从而优化了资源使用和提高了性能。当对象被移动后,其状态变为有效但未定义,这允许编译器进行更高效的优化。
完美转发:通过模板和引用折叠,可以完美地将左值或右值转发到另一个函数,保持其左值或右值的属性。
int a = 10; // a是左值
int b = a; // 赋值操作,a是左值
int func() { return 42; }
int c = func(); // func()的返回值是右值,赋值给c
// C++11及以后
std::string s1 = "hello"; // s1是左值
std::string s2 = std::move(s1); // 使用std::move将s1转换为右值引用,允许“移动”资源
// s1现在处于有效但未定义的状态
核心就是一句话,能取到地址的是左值,取不到地址的是右值。