一、深拷贝与浅拷贝
浅拷贝:简单的赋值拷贝操作
深拷贝:在堆区重新申请空间
由于栈上的数据先进后出,所以p2后释放,在执行析构代码时,新建的堆区数据就被释放,再当p1进行释放时,由于堆区数据已经被释放,导致内存重复释放,代码出错。---浅拷贝
我们通过自己实现一个拷贝函数,即在堆区重新开辟一块数据分配给p2,当p2释放掉新开辟的数据时,p1再进行释放原来的数据,代码成功运行。----深拷贝
代码示例如下:
#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){cout << "Person的拷贝构造函数调用" << endl;m_Age=p.m_Age;//m_Height = p.m_Height;//编译器默认实现就是这行代码//深拷贝操作 重新在堆区再创建一块内存m_Height = new int(*p.m_Height);//解引用出160这个数据}~Person(){//析构代码,将堆区开辟的数据做释放操作if (m_Height != NULL){delete m_Height;m_Height = NULL;}cout << "Person的析构函数调用" << endl;}int m_Age;//年龄int* m_Height;//身高
};void test()//栈上的数据先进后出,所以p2先释放
{Person p1(18,160);cout << "p1的年龄为:"<< p1.m_Age <<"身高为:"<< *p1.m_Height << endl;Person p2(p1);cout << "p2的年龄为:" << p2.m_Age << "身高为:" << *p2.m_Height << endl;
}int main()
{test();return 0;
}
总结:如果属性有在堆区开辟的,一定要自己提供拷贝构造(深拷贝)函数,防止浅拷贝带来的问题
二、初始化列表
作用: C++提供了初始化列表语法,用来初始化属性
语法: 构造函数():属性1(值1),属性2(值2)... {}
#include<iostream>
using namespace std;//初始化列表
class Person
{
public://传统初始化操作//Person(int a, int b, int c)//{// m_A = a;// m_B = b;// m_C = c;//}//初始化列表初始化属性//Person() :m_A(10), m_B(20), m_C(30)//{////}Person(int a,int b,int c) :m_A(a), m_B(b), m_C(c){}int m_A;int m_B;int m_C;
};void test()
{//Person p(10, 20, 30);//Person p;//只能赋值10 20 30不是很灵活Person p(30,20,10);//随意赋值cout << "m_A= " << p.m_A << endl;cout << "m_B= " << p.m_B << endl;cout << "m_C= " << p.m_C << endl;
}
int main()
{test();return 0;
}