左值有明确地址可赋值,右值为临时值不可取地址;C++11引入将亡值支持移动语义,右值引用T&&实现资源窃取避免拷贝,std::move将左值转为右值引用但不保证移动构造被调用。

在C++中,左值(lvalue)和右值(rvalue)是表达式的两种基本分类,它们决定了表达式能否被取地址、能否被赋值等行为。理解值类别对于掌握现代C++的移动语义和完美转发至关重要。
左值与右值的基本概念
左值是指那些有明确内存地址、生命周期较长的表达式。它可以出现在赋值操作的左侧,通常表示一个持久的对象。
例如:
int a = 10;a = 20; // a 是左值
这里的变量 a 是一个典型的左值,它有名字、可以取地址(&a 合法),并且可以在多条语句中使用。
立即学习“C++免费学习笔记(深入)”;
右值则是临时的、即将销毁的值,通常不能取地址,也不能出现在赋值操作的左边。
例如:
int b = a + 5; // a + 5 是右值int c = 42; // 42 是右值
表达式 a + 5 和字面量 42 都是右值——它们没有名字,生命周期短暂,通常只用于初始化或计算。
C++11后的值类别扩展
C++11将值类别细化为五种:左值(lvalue)、纯右值(prvalue)、将亡值(xvalue)、泛左值(glvalue)和右值(rvalue)。其中最重要的是引入了将亡值(xvalue),它表示即将被移走资源的对象。
将亡值通常由std::move()产生:
std::string s1 = "hello";std::string s2 = std::move(s1); // std::move(s1) 是将亡值
这里 std::move(s1) 并不真正移动数据,而是把左值 s1 转换为右值引用类型,使其能够绑定到移动构造函数上。
右值引用与移动语义
右值引用(T&&)是C++11引入的新特性,专门用来绑定右值,尤其是将亡值,从而实现移动语义。
例如:
class MyString {public:
MyString(MyString&& other) noexcept {
data = other.data;
other.data = nullptr;
}
private:
char* data;
};
这个移动构造函数接收一个右值引用 other,可以直接“窃取”其内部资源,避免深拷贝,提升性能。
右值引用的关键作用包括:
- 启用移动语义,减少不必要的拷贝
- 支持完美转发(结合模板和std::forward)
- 让临时对象的资源得以复用
常见误区与使用建议
不要认为所有看起来像“值”的都是右值。比如函数返回对象引用时,结果可能是左值:
std::vectorauto& front = vec.front(); // front 是左值,即使它是“获取来的”
另外,虽然 std::move() 返回右值引用,但它本身并不保证调用移动构造函数——目标类型必须提供移动操作才会生效,否则仍会调用拷贝构造。
使用建议:
- 对大型对象使用 std::move 显式转移所有权
- 在实现类时提供移动构造函数和移动赋值运算符
- 避免对已命名的右值引用变量做误判:即使形如 T&& x,x 本身是左值(因为它有名字)
基本上就这些。掌握左值、右值及其引用机制,是写出高效、现代C++代码的基础。理解它们的行为差异,能帮助你更好利用移动语义和资源管理技巧。










