拷贝构造函数用于初始化新对象为现有对象的副本,其参数为类类型的常量引用。当类含指针成员时,默认拷贝构造函数执行浅拷贝,仅复制指针值,导致多个对象共享同一堆内存,析构时可能引发重复释放或野指针问题。深拷贝则通过自定义拷贝构造函数实现,为新对象分配独立内存并复制内容,确保资源安全。例如,StringHolder类需手动实现深拷贝构造函数、赋值操作符和析构函数,遵循三法则。现代C++推荐使用智能指针或标准库容器以自动管理资源,避免手动内存管理错误。关键在于区分复制指针与复制指针所指内容。

在C++中,拷贝构造函数用于创建一个新对象,并将其初始化为另一个已存在对象的副本。当类中包含指针成员或动态分配资源时,拷贝方式的选择至关重要——这引出了深拷贝与浅拷贝的问题。
什么是拷贝构造函数?
拷贝构造函数是一种特殊的构造函数,其参数是本类类型的常量引用,格式如下:
MyClass(const MyClass& other);当发生以下情况时会自动调用拷贝构造函数:
- 用一个对象初始化另一个对象
- 函数传参时以值传递方式传入对象
- 函数返回对象时以值返回
如果没有显式定义拷贝构造函数,编译器会生成一个默认的版本,执行的是逐成员拷贝,对于指针而言就是简单的地址复制——即浅拷贝。
立即学习“C++免费学习笔记(深入)”;
浅拷贝 vs 深拷贝
浅拷贝只是复制了指针的值(也就是内存地址),两个对象的指针指向同一块堆内存。一旦其中一个对象释放该内存,另一个对象的指针就变成野指针,容易导致程序崩溃或重复释放内存(double free)。
深拷贝则为新对象重新分配一块内存,并将原对象的数据复制过去,两个对象各自拥有独立的资源,互不影响。
举个例子:假设有一个类包含一个指向字符串的指针。
// 错误示范:未定义拷贝构造函数 → 使用默认浅拷贝 // 导致多个对象共享同一块内存
};
如果使用默认拷贝构造函数,两个 StringHolder 对象将共享 data 指向的内存,析构时就会出问题。
如何实现深拷贝构造函数
要正确管理资源,必须自定义拷贝构造函数,进行深拷贝:
class StringHolder { private: char* data; public: StringHolder(const char* str) { if (str) { data = new char[strlen(str) + 1]; strcpy(data, str); } else { data = new char[1]; data[0] = '\0'; } }// 深拷贝构造函数
StringHolder(const StringHolder& other) {
data = new char[strlen(other.data) + 1];
strcpy(data, other.data);
}
~StringHolder() {
delete[] data;
}
// 为安全起见,也应定义赋值运算符(拷贝赋值)
StringHolder& operator=(const StringHolder& other) {
if (this != &other) {
delete[] data; // 释放原有资源
data = new char[strlen(other.data) + 1];
strcpy(data, other.data);
}
return *this;
}};
这样,每个对象都有自己独立的 data 内存空间,避免了资源冲突。
什么时候需要深拷贝?
当你类中的成员涉及动态资源管理时,比如:
- 指针指向堆内存
- 文件句柄、网络连接等系统资源
- STL 容器以外的自定义资源管理结构
就必须手动实现拷贝构造函数、赋值操作符和析构函数——这就是三法则(Rule of Three):如果需要析构函数,通常也需要拷贝构造函数和赋值操作符。
现代C++推荐使用智能指针(如 shared_ptr、unique_ptr)或标准库容器(如 string、vector),它们已经实现了正确的拷贝语义,能自动处理深拷贝逻辑,减少手动管理内存的错误。
基本上就这些。理解深浅拷贝的关键在于搞清楚“复制的是指针本身”还是“复制指针所指向的内容”。只要涉及堆内存,就要警惕浅拷贝带来的风险。









