拷贝构造
-
使用一个已经创建完毕的对象来初始化一个新对象
默认情况下,c++编译器至少给一个类添加3个函数
1.默认构造函数(无参,函数体为空)
2.默认析构函数(无参,函数体为空)
3.默认拷贝构造函数,对属性进行值拷贝
构造函数调用规则如下:
-
如果用户定义有参构造函数,c++不在提供默认无参构造,但是会提供默认拷贝构造
-
如果用户定义拷贝构造函数,c++不会再提供其他构造函数
class Person {
public://无参(默认)构造函数Person() {cout << "无参构造函数!" << endl;}//有参构造函数Person(int a) {age = a;cout << "有参构造函数!" << endl;}//拷贝构造函数Person(const Person& p) {age = p.age;cout << "拷贝构造函数!" << endl;}//析构函数~Person() {cout << "析构函数!" << endl;}
public:int age;
};void test01()
{Person p1(18);//如果不写拷贝构造,编译器会自动添加拷贝构造,并且做浅拷贝操作Person p2(p1);cout << "p2的年龄为: " << p2.age << endl;
}void test02()
{//如果用户提供有参构造,编译器不会提供默认构造,会提供拷贝构造Person p1; //此时如果用户自己没有提供默认构造,会出错Person p2(10); //用户提供的有参Person p3(p2); //此时如果用户没有提供拷贝构造,编译器会提供//如果用户提供拷贝构造,编译器不会提供其他构造函数Person p4; //此时如果用户自己没有提供默认构造,会出错Person p5(10); //此时如果用户自己没有提供有参,会出错Person p6(p5); //用户自己提供拷贝构造
}int main() {test01();system("pause");return 0;
}
个人理解拷贝构造 :就是构造函数里面接收的参数是另一个已经创建完毕的对象Person(const Person& p),然后把另一个对象里面的值复制到自己的对象里面,自己不写系统也会默认提供此函数。
深拷贝与浅拷贝
浅拷贝:简单的赋值拷贝操作
深拷贝:在堆区重新申请空间,进行拷贝操作
如果属性有在堆区开辟的,一定要自己提供拷贝构造函数,防止浅拷贝带来的问题
class Person {
public://无参(默认)构造函数Person() {cout << "无参构造函数!" << endl;}//有参构造函数Person(int age ,int height) {cout << "有参构造函数!" << endl;m_age = age;m_height = new int(height);}//拷贝构造函数 Person(const Person& p) {cout << "拷贝构造函数!" << endl;//如果不利用深拷贝在堆区创建新内存,会导致浅拷贝带来的重复释放堆区问题m_age = p.m_age;m_height = new int(*p.m_height);}//析构函数~Person() {cout << "析构函数!" << endl;if (m_height != NULL){delete m_height;}}
public:int m_age;int* m_height;
};void test01()
{Person p1(18, 180);Person p2(p1);cout << "p1的年龄: " << p1.m_age << " 身高: " << *p1.m_height << endl;cout << "p2的年龄: " << p2.m_age << " 身高: " << *p2.m_height << endl;
}int main() {test01();system("pause");return 0;
}
举例:
Person p1;
Person p2;
拷贝构造 :Person p1;
Person p1(p2);
如果p1类里面的数值没在堆区开辟空间 只是简单的赋值 不会发生什么问题
如果p1类在堆区开辟了空间 p2类没有自己写拷贝构造函数 用的是系统默认的拷贝构造函数
p2类会把p1类里面的东西照搬到自己的类里面 包括p1类里面指向堆区的指针int* m_Height
因为p1和p2里面的 int* m_Height指针指向的是同一块堆区的空间,当执行到析构函数释放内存的时候会把同一块空间释放两次 此时系统就会报错
解决办法就是p2自己开辟一块空间,避免重复释放堆区问题
//拷贝构造函数 Person(const Person& p) {cout << "拷贝构造函数!" << endl;//如果不利用深拷贝在堆区创建新内存,会导致浅拷贝带来的重复释放堆区问题m_age = p.m_age;m_height = new int(*p.m_height);}
调用时机
C++中拷贝构造函数调用时机通常有三种情况
-
使用一个已经创建完毕的对象来初始化一个新对象
-
值传递的方式给函数参数传值
-
以值方式返回局部对象
1.使用一个已经创建完毕的对象来初始化一个新对象
略
2.值传递的方式给函数参数传值
实参传给形参 会拷贝一份一摸一样的给形参 调用拷贝构造函数 形参里面的p怎么改变也不会影响实参(两个p不是一个p)
3.以值方式返回局部对象
person p1创建了一个p1对象 调用默认构造函数 返回p1调用拷贝构造函数 拷贝了一份新的return
然后p1被回收调用析构函数
person p1和 test03里面的p不是一个 p是p1拷贝出来的