1. 概念
在现实生活中,可能存在一个与你一样的自己,我们称其为双胞胎。
那在创建对象时,可否创建一个与一个对象一某一样的新对象呢?
构造函数:只有单个形参,该形参是对本类类型对象的引用(一般常用const修饰),在用已存在的类类型对象创建新对象时由编译器自动调用。
2. 特征
拷贝构造函数也是特殊的成员函数,其特征如下:
1)拷贝构造函数是构造函数的一个重载形式。
2)拷贝构造函数的参数只有一个且必须使用引用传参,使用传值方式会引发无穷递归调用,拷贝函数的参数一直赋值,拷贝工作未完成。
class Date {
public:
Date(int year = 1900, int month = 1, int day = 1) {
_year = year;
_month = month;
_day = day;
}
Date(const Date& d) {
_year = d._year;
_month = d._month;
_day = d._day;
}
private:
int _year;
int _month;
int _day;
};
int main() {
Date d1;
Date d2(d1);
return 0;
}
3)若未显示定义,系统生成默认的拷贝构造函数。 默认的拷贝构造函数对象按内存存储按字节序完成拷贝,对象里有什么就进行拷贝,而资源不会进行拷贝,均指向同一个空间。这种拷贝我们叫做浅拷贝,或者值拷贝。浅拷贝:值拷贝对象本身的内容。深拷贝:拷贝对象本身内容+申请的资源。
class Date {
public:
Date(int year = 1900, int month = 1, int day = 1) {
_year = year;
_month = month;
_day = day;
}
private:
int _year;
int _month;
int _day;
};
int main() {
Date d1;
// 这里d2调用的默认拷贝构造完成拷贝,d2和d1的值也是一样的。
Date d2(d1);
return 0;
}
4)那么编译器生成的默认拷贝构造函数已经可以完成字节序的值拷贝了,我们还需要自己实现吗?当然像日期类这样的类是没必要的。那么下面的类呢?验证一下试试?到底自己需不需要实现拷贝构造主要查看定义的对象有没有申请资源,若定义的对象申请了资源得自己实现拷贝构造,编译器自动实现的仅为浅拷贝。
// 这里会发现下面的程序会崩溃掉?这里就需要我们以后讲的深拷贝去解决。
class String {
public:
String(const char* str = "jack") {
_str = (char*)malloc(strlen(str) + 1);
strcpy(_str, str);
}
~String() {
cout << "~String()" << endl;
free(_str);
}
private:
char* _str;
};
int main() {
String s1("hello");
String s2(s1);
}