拷贝构造函数在C++中用于以同类型对象初始化新对象,典型形式为ClassName(const ClassName& other);它在四种场景下被调用:1. 用一对象初始化另一对象,如MyClass obj2(obj1)或MyClass obj3 = obj1;2. 函数按值传参时复制实参;3. 函数按值返回局部对象,尽管常被返回值优化省略;4. STL容器插入或扩容时复制元素;若类管理动态资源,需实现深拷贝以防浅拷贝错误。

在C++中,拷贝构造函数是一个特殊的构造函数,用于创建一个新对象,并将其初始化为另一个同类型对象的副本。它的典型形式是:
ClassName(const ClassName& other);
当以下几种情况发生时,拷贝构造函数会被调用。理解这些触发时机对掌握资源管理、避免浅拷贝问题至关重要。
1. 对象以另一对象初始化
当你用一个已存在的对象去初始化一个新对象时,拷贝构造函数被调用。
立即学习“C++免费学习笔记(深入)”;
- 直接初始化:如 MyClass obj2(obj1);
- 拷贝初始化:如 MyClass obj3 = obj1;(尽管用了赋值符号,本质仍是构造)
示例代码:
#include iostream>
using namespace std;
class MyClass {
public:
int* data;
MyClass(int val) {
data = new int(val);
cout
}
// 拷贝构造函数
MyClass(const MyClass& other) {
data = new int(*other.data); // 深拷贝
cout
}
~MyClass() {
delete data;
cout
}
};
int main() {
MyClass obj1(10);
MyClass obj2 = obj1; // 调用拷贝构造函数
return 0;
}
2. 函数传参时按值传递对象
当函数参数是类类型的值(而非引用或指针)时,实参会通过拷贝构造函数复制给形参。
例如:
- void func(MyClass obj);
- 调用 func(obj1); 会触发拷贝构造
这可能导致性能开销,尤其是对象较大时。建议使用 const 引用传递来避免不必要的拷贝:void func(const MyClass& obj);
3. 函数返回局部对象(按值返回)
当函数返回一个对象(非引用、非指针)时,通常会调用拷贝构造函数将局部对象复制给接收者。
例如:
MyClass createObject() {
MyClass obj(42);
return obj; // 可能调用拷贝构造(但常被优化)
}
MyClass result = createObject();
注意:现代编译器通常会执行“返回值优化”(RVO)或“拷贝省略”,可能跳过实际的拷贝构造调用。但在语法层面,仍要求拷贝构造函数可访问。
4. 容器操作中的元素插入与扩容
STL容器(如 vector)在扩容或插入元素时,若涉及已有对象的复制,会调用拷贝构造函数。
例如:
- vector.push_back(obj):若容器需要扩容,原有元素会被复制到新内存,触发拷贝构造
- vector 的拷贝构造本身也会复制所有元素
确保自定义类支持正确的拷贝语义,否则可能出现悬空指针或重复释放等问题。
基本上就这些常见场景。只要记住:凡是“从一个对象生成另一个新对象”的过程,且没有使用移动语义,就很可能触发拷贝构造函数。在涉及动态资源管理时,务必实现深拷贝版本的拷贝构造函数,或考虑禁用拷贝(使用 delete)并启用移动语义。不复杂但容易忽略。










