std::move仅是将左值转为右值引用的类型转换,不执行移动操作;它用于转移资源所有权、实现移动构造/赋值函数、向只接受右值的接口传参,使用时需注意noexcept、移动后状态及避免对const对象误用。

std::move 不是移动,只是类型转换
std::move 本身不执行任何移动操作,它只是把一个左值强制转换为右值引用类型(T&&),从而让后续的移动构造函数或移动赋值运算符有机会被调用。如果你的对象没有定义移动构造函数或移动赋值运算符,std::move 后仍会走拷贝——这常被误认为“移动失败”。
常见错误现象:
- 对
int、double等内置类型调用std::move,编译通过但毫无意义 - 移动后继续使用原对象(如访问成员、再次移动),结果未定义(除非你手动置空)
- 在返回局部变量时盲目加
std::move,反而抑制了 NRVO(命名返回值优化)
什么时候该用 std::move:三类典型场景
真正需要 std::move 的地方,基本逃不开这三种情况:
-
转移资源所有权:比如把一个
std::vector的内容交给另一个容器,避免深拷贝 -
实现移动构造/赋值函数:在用户自定义类型的移动构造函数中,用
std::move转移成员 -
向只接受右值的接口传参:例如
std::thread构造时传入可调用对象,或std::unique_ptr的构造/赋值
示例(转移 vector 内容):
立即学习“C++免费学习笔记(深入)”;
std::vectorv1 = {1, 2, 3}; std::vector v2 = std::move(v1); // v1 现在处于有效但未指定状态(通常为空)
左值引用 vs 右值引用:关键在绑定规则,不在名字
左值引用(T&)只能绑定左值;右值引用(T&&)能绑定右值,也能通过引用折叠绑定左值(如模板推导中的 T&& 是万能引用)。这不是语法糖,而是类型系统硬性约束:
-
int x = 42;→x是左值,int& r1 = x;合法,int&& r2 = x;编译失败 -
std::string("hello")是纯右值,std::string&& r = std::string("hello");合法 -
std::move(x)返回int&&,此时可绑定到int&&形参,但不能绑定到int&(除非是 const 左值引用)
容易忽略的坑:移动后状态与 noexcept
标准库中所有移动操作都要求 noexcept(至少是强异常安全),否则容器(如 std::vector 扩容)可能退回到拷贝——这是性能断崖式下降的隐性原因。
- 自定义移动构造函数务必加
noexcept:例如MyClass(MyClass&& other) noexcept; - 移动后原对象必须保持“有效但未指定状态”,不是“不可用”,也不是“必须为空”;你可以检查
v.empty(),但不能假设它一定为 true - 对已声明为
const的对象调用std::move没有意义:const std::string s; std::move(s)得到的是const std::string&&,无法触发非 const 移动函数
std::move,就很容易掉进“以为省了拷贝,其实没动,或者动了却崩了”的坑里。










