复制构造函数用一个已有同类对象创建新对象进行数据初始化:
语法形式:
类名 :: 类名(const 类名 & 引用名 , …);
复制构造函数的特点:
- 复制构造函数名与类名相同,并且也没有返回值类型。
- 复制构造函数可写在类中,也可以写在类外。
- 复制构造函数要求有一个类类型的引用参数。
- 如果没有显式定义复制构造函数,系统自动生成一个默认形式的复制构造函数。
复制构造函数的调用 :
- 声明语句中用类的一个已知对象初始化该类的另一个对象时。
Point ob1;
Point ob2(ob1);
但是:下面的因为s2已提前声明,所以不会调用复制构造函数:
Point ob1;
Point ob2;
ob2=ob1;
- 当对象作为一个函数实参传递给函数的形参时,需要将实参对象去初始化形参对象时,需要调用复制构造函数。
fun(Point ob1);
- 当对象是函数的返回值时,由于需要生成一个临时对象作为函数返回结果,系统需要将临时对象的值初始化另一个对象,需要调用复制构造函数。
Point fun()
{
Point ob2(2,6);
return ob2;
}
深复制:
关于深复制:
●通过一个对象初始化另一个对象时,不仅复制了数据成员,也复制了资源的复制方式称为深复制。
●自定义复制构造函数所进行的复制是浅复制。
定义支持深复制的复制构造函数:
深复制构造函数必须显式定义
深复制构造函数的特点:
- 定义:类名::类名([const] 类名 &对象名);
- 成员变量的处理:对复杂类型的成员变量,使用new操作符进行空间的申请,然后进行相关的复制操作
class String {
private:
char *m_str;
public:
String(char*str = NULL)
{
m_str = new char[strlen(str)+1];
strcpy(m_str, str);
}
String(String& str)
{
m_str = new char[strlen(str.m_str)+1];//深复制,先进行空间分配
strcpy(m_str, str.m_str);
}
~String()
{
if (m_str)
delete[]m_str;
}
};
int main()
{
String s1("Hello");
String s2(s1);
}
传参数的时候不传对象而是传引用
传参的时候加上const修饰,能够提高程序的安全性,鲁棒性,防止意外在函数内部将_stdt进行了改变,影响了结果
若定义构造函数若未显示定义,系统会默认缺省的拷贝构造函数。缺省的拷贝构造函数会依次拷贝类成员进行初始化。不过默认的拷贝构造函数只是进行了 “浅复制”
直接用对象进行传参的时候,首先会在栈帧中开辟新空间存放建立的临时变量,来将实参的内容拷贝进去,而建立了临时变量,在将实参的内容(即对象的内容)拷贝过程中,又要再一次的调用拷贝构造函数,这样的话,会陷入无穷递归当中。