拷贝构造函数在传入左值或需复制对象时调用,如用已有对象初始化新对象、值传递参数、返回局部对象(无RVO)及容器扩容;若未显式定义,编译器生成默认浅拷贝版本。移动构造函数在传入右值(临时对象、std::move结果)时触发,用于窃取资源以避免深拷贝,如初始化于临时对象、返回匿名对象、std::move转换或容器插入即将失效对象;仅当类声明移动操作时才启用移动语义,否则退化为拷贝。优先级上,左值调用拷贝,右值调用移动;若无移动构造,则右值也使用拷贝构造。c++11后若定义移动构造或赋值,编译器不再自动生成拷贝构造,需遵循“三五法则”或“零一法则”管理资源。

在C++中,拷贝构造函数和移动构造函数用于对象的初始化过程,它们的调用时机取决于传入参数的值类别(左值或右值)以及类是否显式定义了这些函数。理解它们的触发条件对编写高效代码至关重要。
拷贝构造函数的调用时机
拷贝构造函数在以下情况被调用,前提是传入的是一个左值或者需要复制的对象:
- 用一个已存在的对象初始化新对象,例如:MyClass obj2(obj1); 或 MyClass obj2 = obj1;
- 函数参数以值传递方式传入对象时,会复制实参
- 函数返回一个局部对象,且未启用返回值优化(RVO/NRVO)时,可能调用拷贝构造(现代编译器通常优化掉)
- 对象被插入容器(如vector扩容)时,已有元素需要复制
注意:如果类中没有显式定义拷贝构造函数,编译器会自动生成默认的,执行逐成员的浅拷贝。
移动构造函数的调用时机
移动构造函数用于“窃取”临时对象或即将销毁对象的资源,避免不必要的深拷贝,它在以下场景被触发:
- 用一个右值(临时对象)初始化新对象,例如:MyClass obj2(std::move(obj1)); 或 MyClass obj2(MyClass());
- 函数返回一个匿名临时对象(非引用),且不被优化时,可能调用移动构造
- 通过 std::move 将左值显式转换为右值引用后进行初始化
- 容器操作中(如vector.push_back(std::move(obj)))插入即将失效的对象
注意:只有当类中声明了移动构造函数(或移动赋值运算符)时,编译器才不会禁用移动语义。否则,即使有右值传入,仍可能退化为拷贝。
优先级与隐式行为
当同时存在拷贝和移动构造函数时,编译器根据参数的值类别决定调用哪一个:
立即学习“C++免费学习笔记(深入)”;
- 传入左值 → 调用拷贝构造
- 传入右值(临时对象、std::move结果)→ 调用移动构造
- 如果没有移动构造函数,右值也会通过拷贝构造处理
- 如果类定义了移动构造或移动赋值,编译器不会自动生成拷贝构造函数(C++11后规则)
基本上就这些。掌握值类别与构造函数匹配逻辑,能更好利用移动语义提升性能,避免意外拷贝。实践中建议遵循“三五法则”或“零一法则”管理资源。


