0

0

c++怎么利用std::move避免不必要的内存拷贝_c++ 移动语义与性能优化【教程】

尼克

尼克

发布时间:2026-01-07 16:43:17

|

603人浏览过

|

来源于php中文网

原创

std::move仅是将左值转为右值引用的类型转换,不执行移动操作;若对象无移动语义则退化为拷贝;仅对管理堆内存的类型有意义,移动后原对象处于有效但未指定状态,不可再读取。

c++怎么利用std::move避免不必要的内存拷贝_c++ 移动语义与性能优化【教程】

std::move 不是移动,只是类型转换

std::move 本身不执行任何拷贝或移动操作,它只是一个强制类型转换函数,把左值转成右值引用(T&&),从而让编译器有机会调用移动构造函数或移动赋值运算符。如果你的对象没有定义移动语义(即没写移动构造函数/移动赋值运算符),std::move 后仍会退化为拷贝。

  • 常见错误:对 intstd::array 等 trivial 类型滥用 std::move,毫无收益,还可能干扰编译器优化
  • 只有含堆内存管理的类(如 std::vectorstd::string、自定义容器)才真正受益于移动
  • 移动后原对象处于“有效但未指定状态”,不可再读取其值(比如不能继续访问 v.data()s.c_str()),除非重新赋值

什么时候该显式调用 std::move

典型场景是“你确定这个对象后续不再需要,且它支持移动”——最常见于函数返回、容器插入、资源交接。

  • 函数返回局部对象时,编译器通常会自动应用 RVO(返回值优化),此时 std::move 反而阻止优化,应避免:
    std::vector make_data() {
        std::vector v(1000000);
        return v; // ✅ 让编译器自己决定;加 std::move(v) 是错的
    }
  • 向容器末尾移动插入大对象时,必须用:
    std::vector vec;
    std::string s = "very long string...";
    vec.push_back(std::move(s)); // ✅ 避免拷贝字符串内部 buffer
  • 实现移动赋值运算符时,需对成员逐个 std::move
    MyClass& operator=(MyClass&& other) noexcept {
        if (this != &other) {
            data_ = std::move(other.data_); // ✅ 移动内部 vector
            id_ = std::exchange(other.id_, 0); // 其他标量可直接赋值或用 exchange
        }
        return *this;
    }

std::move 后访问原对象的典型崩溃

移动后误用原对象是运行时隐患,尤其在调试通过但 Release 崩溃。例如:

std::vector v = {1,2,3};
auto&& w = std::move(v);
std::cout << v.size(); // ❌ 未定义行为:v 已被掏空,size() 可能返回 0、垃圾值,甚至 segfault
  • 不要假设移动后原对象“清零”或“变为空”——标准只保证“可析构、可赋值”,具体状态由类实现决定
  • 调试时可用 AddressSanitizer 捕获部分越界访问,但无法检测所有逻辑误用
  • 若需保留原对象语义,改用 std::swap 或显式复制

移动语义生效的前提条件

即使写了 std::move,移动也未必发生。以下任一条件不满足,就会回退到拷贝:

360智图
360智图

AI驱动的图片版权查询平台

下载

立即学习C++免费学习笔记(深入)”;

  • 目标类型未声明移动构造函数(T(T&&))或移动赋值(T& operator=(T&&)
  • 移动操作被 delete 或非 noexcept(某些 STL 容器如 std::vector::resize 要求移动 noexcept 才敢用)
  • 源对象是 const 左值(const std::string s;),std::move(s) 得到的是 const T&&,无法绑定到非 const 移动函数
  • 编译器开启优化(如 -O2)后,某些拷贝本就被 elision 掉,std::move 反而干扰优化路径

检查是否真触发了移动,最直接方式是给类加上带日志的移动构造函数,或用 std::is_move_constructible_v 编译期断言。

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

315

2023.08.02

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1462

2023.10.24

Go语言中的运算符有哪些
Go语言中的运算符有哪些

Go语言中的运算符有:1、加法运算符;2、减法运算符;3、乘法运算符;4、除法运算符;5、取余运算符;6、比较运算符;7、位运算符;8、按位与运算符;9、按位或运算符;10、按位异或运算符等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

227

2024.02.23

php三元运算符用法
php三元运算符用法

本专题整合了php三元运算符相关教程,阅读专题下面的文章了解更多详细内容。

85

2025.10.17

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

520

2023.09.20

java进行强制类型转换
java进行强制类型转换

强制类型转换是Java中的一种重要机制,用于将一个数据类型转换为另一个数据类型。想了解更多强制类型转换的相关内容,可以阅读本专题下面的文章。

282

2023.12.01

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

315

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

529

2024.08.29

java学习网站推荐汇总
java学习网站推荐汇总

本专题整合了java学习网站相关内容,阅读专题下面的文章了解更多详细内容。

6

2026.01.08

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 6.2万人学习

C 教程
C 教程

共75课时 | 3.9万人学习

C++教程
C++教程

共115课时 | 11.5万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号