右值引用(T&&)允许绑定临时对象,移动构造函数(ClassName(ClassName&&))通过转移资源避免深拷贝,提升性能。

在C++11中,移动语义和右值引用是提升性能的关键机制,尤其在处理临时对象和资源管理时。它们的核心目标是避免不必要的深拷贝,通过“移动”而非“复制”来转移资源,比如动态内存、文件句柄等。
右值引用与&&符号
右值引用使用T&&语法,绑定到临时对象(右值),这些对象即将被销毁,因此可以安全地“窃取”其内部资源。
普通引用(左值引用)如T&只能绑定到具名对象,而右值引用能捕获那些无法取地址的临时值。
例如:
立即学习“C++免费学习笔记(深入)”;
string createTemp() { return "hello"; }string&& temp = createTemp(); // 合法:绑定到临时string对象
移动构造函数的作用
移动构造函数的形式为:ClassName(ClassName&& other),它接收一个右值引用参数,把原对象的资源“移动”到新对象,同时将原对象置于有效但可析构的状态。
典型实现如下:
class MyString {char* data;
public:
MyString(MyString&& other) noexcept {
data = other.data; // 转移指针
other.data = nullptr; // 防止原对象释放资源
}
};
这样,构造新对象时无需分配新内存和复制内容,直接拿走临时对象的数据即可。
资源转移的实际场景
当函数返回一个局部对象时,编译器会尝试将其作为右值处理,触发移动构造而不是拷贝构造。
如果没有定义移动构造函数,系统会回退到拷贝构造,造成性能浪费。
常见受益类型包括:
- std::vector:移动时只复制指针和大小,不复制整个缓冲区
- std::unique_ptr:所有权直接转移,无需删除旧资源
- 临时字符串拼接结果:避免中间结果的多次内存分配
注意:移动操作应标记为noexcept,否则某些STL操作(如vector扩容)可能拒绝使用移动而选择更安全的拷贝。
完美转发与std::move
std::move不是真正“移动”,而是将左值强制转换为右值引用,以便调用移动构造或移动赋值函数。
例如:
立即学习“C++免费学习笔记(深入)”;
MyString a = "world";MyString b = std::move(a); // a的内容被“掏空”,b接管资源
此后a仍可析构,但不应再用于读取数据。
基本上就这些。移动语义不复杂,但容易忽略细节。只要记住:右值引用让你识别“可掠夺”的对象,移动构造让掠夺合法且高效。











