深拷贝
深拷贝:在堆区重新申请空间,进行拷贝操作
浅拷贝
浅拷贝:简单的复制拷贝操作
class person
{
public:
person()
{
cout<<"默认构造函数调用"<<endl;
}
person(int age)
{
m_age = age;
cout<<"默认构造函数调用"<<endl;
}
`person()
{
cout<<"析构函数调用"<<endl;
}
int m_age;
}
void test()
{
person p1(18);
cout<<"p1的年龄为:"<<p1.m_age<<endl;//输出的年龄应该为12
person p2(p1);//调用拷贝构造函数来创建p2
cout<<"p2的年龄为:"<<p2.m_age<<endl;//输出为12,编译器调用浅拷贝函数。
}
当参数中有堆区参数,改造之后:
class person
{
public:
person()
{
cout<<"默认构造函数调用"<<endl;
}
person(int age,int height)
{
m_age = age;
m_height = new int(height);
cout<<"默认构造函数调用"<<endl;
}
`person()
{
//析构代码,将堆区开辟数据做释放操作
if(m_height!= NULL)
{
delete m_height;
m_height = NULL:
}
cout<<"析构函数调用"<<endl;
}
int m_age;
int *m_height;//身高 用指针是为了开辟到堆区
}
void test()
{
person p1(18,160);
cout<<"p1的年龄为:"<<p1.m_age<<"身高为:"<<*p1.m_height<<endl;//输出的年龄应该为12
person p2(p1);//调用拷贝构造函数来创建p2
cout<<"p2的年龄为:"<<p2.m_age<<"身高为:"<<*p2.m_height<<endl;//输出的年龄应该为12
}
上面这个代码运行完成后,将会报错,报:对象的初始化和清理。
因为对于age这个属性,当person p2(p1)时,编译器会调用自己的拷贝构造函数,会做浅拷贝操作,会逐字节的将p1的元素拷贝到p2中。但是在堆区的height属性,他同样会将堆区的地址拷贝到p2。当p1和p2执行析构函数时,p2会率先被释放,当p1释放时,发现该内存已经被释放过了,自己再操作这块内存就属于非法操作,浅拷贝带来的问题就是:堆区的内存重复释放。
解决办法:
利用深拷贝来解决他。重写拷贝构造函数,重新在堆区创建空间。
//重写拷贝构造函数,实现深拷贝
person(const person &p)
{
cout<<"拷贝构造函数调用"<<endl;
m_age=p.m_age;
//m_height = p.m_height; 编译器默认实现就是这行代码
//深拷贝操作
m_height = new int(*p.m_height);
}