在看以前的笔记时发现关于浅拷贝与深拷贝的知识点整理的有点模糊。所以把内容重新整理一下。
简单来说,
浅拷贝(位拷贝):简单的赋值拷贝操作。类中的合成拷贝构造函数就是浅拷贝。
深拷贝:在堆区重新申请空间,进行拷贝操作。
先说总结:如果属性有在堆区开辟的,一定要自己提供拷贝构造函数,防止浅拷贝带来问题。如,若用合成拷贝构造函数来进行拷贝初始化,则在调用析构函数释放自身在堆区开辟的内存空间时,会导致堆区内存被重复释放,致使程序崩溃。
现用一段示例代码进一步说明该结论:
#include <iostream>
using namespace std;
class Person
{
public:
Person() {
cout << "Person默认构造函数调用" << endl;
}
Person(int age, int height) {
m_Age = age;
m_Height = new int(height);
cout << "Person有参构造函数调用" << endl;
}
//自己实现拷贝构造函数,解决浅拷贝带来的问题
Person(const Person& p) {
m_Age = p.m_Age;
cout << "Person拷贝构造函数调用" << endl;
//m_Height =p.m_Height;编译器默认实现就是这行代码;
//深拷贝操作
m_Height = new int(*p.m_Height);
}
~Person() {
//析构代码,将堆区开辟数据做释放操作
cout << "Person析构函数调用" << endl;
if (m_Height != NULL) {
delete m_Height;
//避免出现野指针,将其置空。
m_Height = NULL;
}
}
public:
int m_Age = 10;
int * m_Height =NULL;
};
void test01() {
Person p1(16,170);
cout << "p1的年龄为:" << p1.m_Age <<" p1的身高为:"<<*p1.m_Height<<endl;
Person p2(p1);
cout << "p2的年龄为:" << p2.m_Age <<" p2的身高为:"<<*p2.m_Height<<endl;
}
int main()
{
test01();
return 0;
}
再用两张图来进一步说明:
第一张是使用合成拷贝构造函数:注意两个指针指向同一堆空间
要解决这个问题,自己创建一个拷贝构造函数,在堆区重新申请一块内存,存放相同的数据m_Height,此时p1和p2的m_Height指向的内存空间已经不一样了。