当前位置: 首页 > web >正文

std::move 和 std::forward

关联点

都是执行转换(cast)的函数(函数模板),不产生任何可执行代码。且都可以把实参转换成右值。
std::move无条件将实参(const除外 )转换成右值引用,std::forward 条件返回右值引用


_EXPORT_STD template <class _Ty>
_NODISCARD _MSVC_INTRINSIC constexpr remove_reference_t<_Ty>&& move(_Ty&& _Arg) noexcept {return static_cast<remove_reference_t<_Ty>&&>(_Arg);
}

_EXPORT_STD template <class _Ty>
_NODISCARD _MSVC_INTRINSIC constexpr _Ty&& forward(remove_reference_t<_Ty>& _Arg) noexcept {return static_cast<_Ty&&>(_Arg);
}_EXPORT_STD template <class _Ty>
_NODISCARD _MSVC_INTRINSIC constexpr _Ty&& forward(remove_reference_t<_Ty>&& _Arg) noexcept {static_assert(!is_lvalue_reference_v<_Ty>, "bad forward call");return static_cast<_Ty&&>(_Arg);
}

_EXPORT_STD template <class _Ty>
struct remove_reference {using type                 = _Ty;using _Const_thru_ref_type = const _Ty;
};template <class _Ty>
struct remove_reference<_Ty&> {using type                 = _Ty;using _Const_thru_ref_type = const _Ty&;
};template <class _Ty>
struct remove_reference<_Ty&&> {using type                 = _Ty;using _Const_thru_ref_type = const _Ty&&;
};_EXPORT_STD template <class _Ty>
using remove_reference_t = typename remove_reference<_Ty>::type;

先去引用,然后转换成右值

区别

1. 用途不同

  • std::move
    无条件将左值转换为右值引用,表示对象资源可被“移动”。用于触发移动构造函数或移动赋值运算符,避免深拷贝。
    示例:
std::string s1 = "Hello";
std::string s2 = std::move(s1); // s1 的资源被移动到 s2,s1 不再有效
  • std::forward
    有条件地保持值类别(左值/右值),用于完美转发。通常与万能引用(T&&)配合,在泛型代码中保持参数的原始类型。
    示例:
template<typename T>
void wrapper(T&& arg) {callee(std::forward<T>(arg)); // 保持 arg 的原始值类别
}

2. 转换条件

  • std::move
    无论输入是左值还是右值,始终返回右值引用。
auto rval = std::move(lval); // 无条件转为右值
  • std::forward
    根据模板参数 T 的类型决定转换行为:
forward<T>(arg); // 类型 T 决定结果
- 若 `T` 是左值引用(如 `int&`),返回左值引用。  
- 若 `T` 是非引用或右值引用(如 `int` 或 `int&&`),返回右值引用。

3. 实现机制

  • std::move** 实现**
template<typename T>
constexpr typename std::remove_reference<T>::type&& move(T&& t) noexcept {return static_cast<typename std::remove_reference<T>::type&&>(t);
}

直接通过 static_cast 将输入转换为右值引用。

  • std::forward** 实现**
template<typename T>
constexpr T&& forward(typename std::remove_reference<T>::type& t) noexcept {return static_cast<T&&>(t);
}

根据 T 的类型推断结果,决定返回左值或右值引用。


4. 应用场景

  • 使用 std::move 的场景
    • 需要明确转移对象资源所有权时(如实现移动构造函数)。
    • 避免拷贝开销,例如将局部变量移动到容器中:
std::vector<std::string> vec;
std::string s = "data";
vec.push_back(std::move(s)); // 移动而非拷贝
  • 使用 std::forward 的场景
    • 泛型函数模板中转发参数,保持其原始值类别:
template<typename... Args>
void emplace(Args&&... args) {container.emplace_back(std::forward<Args>(args)...); // 完美转发参数包
}

总结

特性std::movestd::forward
目的强制转为右值,触发移动语义保持参数原始值类别,完美转发
转换条件无条件依赖模板参数 T
典型应用移动构造函数、避免拷贝泛型代码中的参数转发
参数类型接受任意类型通常与万能引用 T&& 配合使用

补充:

//拷贝构造

MyStruct(const Mystruct &)

{

}

//移动构造

Mystruct( Mystruct &&) // 无const

{

}
万能引用:
T&&

http://www.xdnf.cn/news/5792.html

相关文章:

  • 图像的EXIF方向信息(Orientation标签)
  • MySQL 备份与恢复
  • 45、跳跃游戏Ⅱ
  • JavaScript双问号操作符(??)详解,解决使用 || 时因类型转换带来的问题
  • 消息队列RocketMQ-docker部署保姆级教程(从0到1)(2)
  • 16.three官方示例+编辑器+AI快速学习webgl_buffergeometry_lines_indexed
  • apt 软件源与 Docker 镜像源
  • Westlake-Omni 情感端音频生成式输出模型
  • 软考高分备考秘籍:综合知识、案例分析、论文全攻略
  • 如何使用VBA宏高效操作Word文档中的表格(对齐与样式)
  • 六、STM32 HAL库回调机制详解:从设计原理到实战应用
  • nginx-整合modsecurity做waf
  • Ubuntu 22初始配置(root、ssh)
  • 航电系统之电传飞行控制系统篇
  • IDR方程迭代求解算法介绍与比较
  • Ollama+OpenWebUI+docker完整版部署,附带软件下载链接,配置+中文汉化+docker源,适合内网部署,可以局域网使用
  • Java 线程的堆栈跟踪信息
  • 《Python星球日记》 第62天:图像方向综合项目(猫狗分类)
  • Java自动化测试
  • 2025年5月13日 奇门遁甲与股市
  • 学习笔记:黑马程序员JavaWeb开发教程(2025.4.3)
  • 麒麟 v10 卸载podman
  • 【大模型MCP协议】MCP官方文档(Model Context Protocol)一、开始——1. 介绍
  • pythonocc 拉伸特征
  • C语言 第六章 结构体(3)
  • 0前言(文章体系)
  • 数字滤波器应用介绍
  • 流体力学绪论(期末复习)
  • 【android bluetooth 框架分析 02】【Module详解 13】【CounterMetrics 模块介绍】
  • 继承关系下创建对象的具体流程