返回局部对象时编译器通常已启用RVO/NRVO,直接返回obj即可;加std::move反而禁用优化、强制调用移动构造函数,得不偿失。
std::move
直接结论:在函数中返回一个局部对象(如 return obj;),现代C++编译器(GCC 5+、Clang 3.5+、MSVC 2015+)默认启用返回值优化(RVO)或命名返回值优化(NRVO),会直接在调用方的内存位置构造对象,**完全绕过拷贝或移动**。此时加 std::move 不仅没收益,反而可能阻止RVO——因为 std::move(obj) 把左值转成右值,让编译器无法识别这是可优化的“具名局部对象”,被迫退化为调用移动构造函数。
std::move 后,该变量变成右值表达式,NRVO失效,强制触发移动构造MyClass create_object() {
MyClass obj;
// ... 初始化
return obj; // ✅ 推荐:让编译器决定是否RVO
// return std::move(obj); // ❌ 不推荐:禁用NRVO,强制移动
}
std::move?看对象是不是“可移动的右值”显式 std::move 的合理场景,是当你持有某个左值(比如函数参数、成员变量、局部变量),又明确知道它后续不再使用,且你想把它“移交
”给另一个对象时。典型如实现移动构造函数或移动赋值运算符:
std::move 将参数的成员“搬”到新对象:如 data_(std::move(other.data_))
vec.push_back(std::move(temp_obj));
return std::move(param_obj);
MyClass(MyClass&& other) noexcept
: data_(std::move(other.data_)), // ✅ 必须:转移内部资源
id_(std::exchange(other.id_, 0)) {}
std::move本身不移动任何东西,它只是类型转换工具
std::move 只是一个强制类型转换函数,签名是 template,它不执行任何内存操作,也不调用任何构造函数。它的唯一作用是把一个左值表达式(如变量名)转成右值引用类型,从而让后续的重载决议选中移动构造/移动赋值版本。
std::move(x) 仍会退化为调用拷贝构造(因为右值也能绑定到 const T&)int、double)或 trivial 类型调用 std::move 没有意义,编译器会直接按值传递std::move 后继续访问原对象(未定义行为):如 auto x = std::move(obj); use(obj); —— obj 此时处于有效但未指定状态不要靠“感觉”或打印构造函数日志判断RVO——因为即使你写了拷贝/移动构造函数,只要它们有副作用(如std::cout ),编译器就可能因“需要保留可观测行为”而禁用RVO(C++17起保证某些情况下的强制RVO,但带副作用的构造函数仍是例外)。
-O2 -S 生成汇编,搜索是否出现 call 到拷贝/移动构造函数-fno-elide-constructors),再对比运行时行为真正影响性能的关键,不是纠结某次返回要不要加 std::move,而是确保你的类正确声明了 noexcept 移动操作——这直接影响 std::vector 扩容时能否安全移动而非拷贝元素。