0

0

c++的移动构造函数(Move Constructor)何时会被自动生成? (隐式声明规则)

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-01-14 12:41:02

|

453人浏览过

|

来源于php中文网

原创

编译器仅在类无用户声明的移动/拷贝操作和析构函数、且所有成员及基类均可移动时隐式声明移动构造函数;否则被删除或不声明。

c++的移动构造函数(move constructor)何时会被自动生成? (隐式声明规则)

哪些条件下编译器会隐式声明移动构造函数

只有当类中没有用户声明任何移动操作(移动构造函数或移动赋值运算符),且所有非静态成员和基类都可移动时,C++11 及之后标准才可能隐式声明移动构造函数。

但注意:隐式声明 ≠ 隐式定义。编译器只在需要时(如 std::move 触发、返回局部对象等)才生成定义,且前提是该隐式声明未被删除。

  • 若类声明了拷贝构造函数、拷贝赋值运算符、析构函数中的任一个(且未显式声明移动操作),编译器不会隐式声明移动构造函数(这是 C++11 的“移动操作抑制规则”)
  • 若某个非静态成员类型没有移动构造函数(例如只有拷贝),即使其他条件满足,整个类的移动构造函数也会被隐式定义为 delete
  • 基类若不可移动(如其移动构造函数为 delete 或未声明),派生类也不会获得隐式移动构造函数

如何判断你的类是否拥有隐式声明的移动构造函数

最可靠的方法是用 static_assert 检查 std::is_move_constructible_v,并结合编译器诊断:

struct S {
    std::string s;
    int x;
}; // 无用户定义的特殊成员 → 编译器隐式声明移动构造函数

static_assert(std::is_move_constructible_v); // ✅ 通过

如果出现类似错误:

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

error: use of deleted function 'S::S(S&&)'

说明编译器确实隐式声明了它,但因某成员/基类不可移动而被自动设为 delete

Magician
Magician

Figma插件,AI生成图标、图片和UX文案

下载
  • clang++ -std=c++17 -Xclang -ast-dump | grep "MoveConstructor" 可查看 AST 中是否隐式声明
  • GCC 不直接暴露隐式声明信息,但 -fno-elide-constructors + 调试构造函数调用可间接验证
  • 添加 = default 显式请求(如 S(S&&) = default;)能强制生成,并让错误更早暴露

为什么有时候明明没写移动构造函数,却无法 move

常见原因不是“没生成”,而是“被隐式定义为 delete”。典型场景包括:

  • 类中含 const 成员或引用成员:它们不能被修改,故移动构造函数无法为它们“掏空”原对象
  • 某个成员类型只有拷贝构造函数(如老旧自定义类未加 T(T&&)),导致合成失败
  • 显式删除了拷贝构造函数但没提供移动构造函数:S(const S&) = delete; 本身不阻止移动,但若同时存在用户定义析构函数,就会抑制隐式移动构造函数声明
  • 继承链中某基类的移动构造函数是 privatedelete,派生类无法访问

此时 std::move(x) 仍产生右值引用,但调用移动构造函数时触发 deleted 函数错误。

显式写 = default 和完全不写,行为有区别

有本质区别。不写时,是否隐式声明取决于上述严格规则;而写 T(T&&) = default;显式请求编译器合成,规则更宽松:

  • 即使类有用户定义的析构函数,= default 仍可成功(只要成员/基类可移动)
  • 合成过程会检查每个子对象是否可移动;若不可,该 = default 移动构造函数被定义为 delete,但你明确知道这是你主动要求的
  • 它让意图清晰,避免依赖隐式规则带来的不确定性,也方便后续加日志或断点调试

所以现代 C++ 实践中,只要类语义上支持移动,建议显式写 T(T&&) = default;,而不是赌编译器是否隐式声明。

真正容易被忽略的是:移动构造函数是否被隐式声明,不取决于你“有没有用到”,而取决于类定义本身的结构和成员类型是否满足一整套静默条件——哪怕你从没调用过 move,只要有一个成员拖后腿,它就无声无息地变成 delete

相关专题

更多
java基础知识汇总
java基础知识汇总

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

1463

2023.10.24

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

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

228

2024.02.23

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

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

85

2025.10.17

scripterror怎么解决
scripterror怎么解决

scripterror的解决办法有检查语法、文件路径、检查网络连接、浏览器兼容性、使用try-catch语句、使用开发者工具进行调试、更新浏览器和JavaScript库或寻求专业帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

187

2023.10.18

500error怎么解决
500error怎么解决

500error的解决办法有检查服务器日志、检查代码、检查服务器配置、更新软件版本、重新启动服务、调试代码和寻求帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

271

2023.10.25

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

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

523

2023.09.20

数据库Delete用法
数据库Delete用法

数据库Delete用法:1、删除单条记录;2、删除多条记录;3、删除所有记录;4、删除特定条件的记录。更多关于数据库Delete的内容,大家可以访问下面的文章。

269

2023.11.13

drop和delete的区别
drop和delete的区别

drop和delete的区别:1、功能与用途;2、操作对象;3、可逆性;4、空间释放;5、执行速度与效率;6、与其他命令的交互;7、影响的持久性;8、语法和执行;9、触发器与约束;10、事务处理。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

208

2023.12.29

Java 桌面应用开发(JavaFX 实战)
Java 桌面应用开发(JavaFX 实战)

本专题系统讲解 Java 在桌面应用开发领域的实战应用,重点围绕 JavaFX 框架,涵盖界面布局、控件使用、事件处理、FXML、样式美化(CSS)、多线程与UI响应优化,以及桌面应用的打包与发布。通过完整示例项目,帮助学习者掌握 使用 Java 构建现代化、跨平台桌面应用程序的核心能力。

36

2026.01.14

热门下载

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

精品课程

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

共94课时 | 6.7万人学习

C 教程
C 教程

共75课时 | 4万人学习

C++教程
C++教程

共115课时 | 12.2万人学习

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

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