拷贝构造函数在c++中用于初始化新对象为同类型对象的副本,主要调用场景包括:1. 用对象初始化另一对象,如MyClass obj2 = obj1;2. 值传递传参时复制实参;3. 函数值返回局部对象(尽管可能被RVO优化);4. 对象添加到容器时的值语义操作;5. 显式通过new或直接语法调用。若未定义,编译器生成默认浅拷贝版本,管理动态资源时需自定义实现深拷贝以避免资源问题。

在C++中,拷贝构造函数是一个特殊的构造函数,用于创建一个新对象,并将其初始化为另一个同类型对象的副本。理解拷贝构造函数的调用时机对于掌握对象生命周期和资源管理至关重要。下面总结了拷贝构造函数被调用的主要场景。
1. 用一个对象初始化另一个对象
当使用一个已存在的对象去初始化一个新创建的对象时,会调用拷贝构造函数。
- 例如:MyClass obj2(obj1); 或 MyClass obj2 = obj1;
- 注意:这里的“=”不是赋值操作,而是初始化语法,仍会触发拷贝构造
2. 函数传参时以值传递方式传递对象
当函数的参数是类类型,并且以值传递(而非引用或指针)的方式传入时,实参会通过拷贝构造函数复制给形参。
- 例如:void func(MyClass obj); 调用时 func(instance);
- 此时instance会被拷贝一份传入函数,调用拷贝构造函数
3. 函数返回局部对象时(值返回)
当函数返回一个局部对象,且返回类型为类类型(非引用、非指针)时,通常会调用拷贝构造函数来创建返回值的副本。
立即学习“C++免费学习笔记(深入)”;
- 例如:MyClass getObj() { MyClass a; return a; }
- 虽然现代编译器常通过返回值优化(RVO/NRVO)省略拷贝,但语义上仍视为可能发生拷贝构造
4. 对象放入容器时(如vector)
向标准库容器(如vector、list等)添加对象时,如果使用的是值语义操作(如push_back传值),可能会触发拷贝构造。
- 例如:vec.push_back(obj); 会将obj拷贝进容器
- C++11后可通过move语义避免不必要的拷贝
5. 显式调用拷贝构造函数
程序员可以显式地通过语法直接调用拷贝构造函数创建新对象。
- 例如:MyClass* p = new MyClass(otherObj);
- 这明确要求用otherObj初始化新对象
基本上就这些常见情况。需要注意的是,如果类中没有显式定义拷贝构造函数,编译器会自动生成一个默认的(进行浅拷贝)。对于包含动态资源的类,通常需要自己定义拷贝构造函数以实现深拷贝,防止资源重复释放等问题。


