右值引用(&&)绑定临时对象,std::move将左值转为右值引用以触发移动构造或赋值,避免深拷贝。自定义类需实现noexcept的移动构造和赋值函数,确保资源安全转移。常见于返回局部对象、插入容器等场景,使用后原对象不可再用。

在C++中,std::move 和 右值引用 是实现移动语义的核心机制,能够显著提升性能,避免不必要的深拷贝。理解它们的用法和原理对写出高效C++代码至关重要。
右值引用的基本概念
右值引用使用 && 语法声明,用来绑定临时对象(右值)。与左值引用(&)不同,右值引用可以修改所绑定的对象,并通常用于“窃取”资源。
例如:
std::string createString() {
return "hello";
}
std::string&& rref = createString(); // 绑定到临时对象
这里 createString() 返回一个临时 std::string 对象,它是一个右值,只能由右值引用接收。
立即学习“C++免费学习笔记(深入)”;
std::move 的作用
std::move 并不真正“移动”任何东西,它只是一个类型转换函数:将一个左值强制转换为右值引用,从而允许调用移动构造函数或移动赋值操作符。
示例:
std::string a = "world";
std::string b = std::move(a); // 调用 string 的移动构造函数
执行后,a 处于合法但未定义状态(通常为空),b 拿走了原本属于 a 的资源。这比拷贝字符串内容要快得多。
自定义类中的移动语义
为了让自己的类支持移动操作,需要显式定义移动构造函数和移动赋值运算符。
例子:
class MyVector {
private:
int* data;
size_t size;
public:
// 移动构造函数
MyVector(MyVector&& other) noexcept
: data(other.data), size(other.size) {
other.data = nullptr;
other.size = 0;
}
// 移动赋值运算符
MyVector& operator=(MyVector&& other) noexcept {
if (this != &other) {
delete[] data;
data = other.data;
size = other.size;
other.data = nullptr;
other.size = 0;
}
return *this;
}
};
关键点:
- 参数是右值引用 MyVector&&
- 使用 noexcept 标记,确保标准库容器能安全使用移动操作
- 原对象资源被“掏空”,防止析构时重复释放
何时使用 std::move
常见使用场景包括:
- 将局部对象从函数返回(编译器常自动优化,但可显式 move)
- 插入容器时避免拷贝:vec.push_back(std::move(obj))
- 交换资源或转移所有权
- 构建临时对象传递给函数
注意:一旦对某个对象使用了 std::move,就不要再使用它,除非重新赋值。
基本上就这些。掌握右值引用和 std::move,能让C++程序更高效、更现代。不复杂但容易忽略细节,比如异常安全和资源清理。










