目录
- 拷贝构造函数
- 一、为什么用拷贝构造
- 二、拷贝构造函数
- 1、概念
- 2、特征
- 1. 拷贝构造函数是构造函数的一个重载形式。
- 2. 拷贝构造函数的参数
- 3. 若未显式定义,编译器会生成默认的拷贝构造函数。
- 4. 拷贝构造函数典型调用场景
 
 
 
拷贝构造函数
一、为什么用拷贝构造
日期类传值(这里是浅拷贝)
#include<iostream>
using namespace std;
class Date {
public:
	Date(int year=1, int month=1, int day=1) {
		_year = year;
		_month = month;
		_day = day;
	}
	void Print() {
		cout << _year << "/" << _month << "/" << _day << endl;
	}
private:
	int _year ;
	int _month ;
	int _day ;
};
int main() {
	Date d1;
	d1.Print();
	return 0;
}
运行后:
 
这里进行了传值的拷贝,形参传给实参,进行了值拷贝,也就是浅拷贝,所以并没有出现问题。
但是栈类的结构浅拷贝会出现问题
#include<iostream>
using namespace std;
class Stack
	{
	public:
		Stack(size_t capacity = 3)
		{
			cout << "Stack(size_t capacity = 3)" << endl;
			_a = (int*)malloc(sizeof(int) * capacity);
			if (nullptr == _a)
			{
				perror("malloc申请空间失败!!!");
			}
			_capacity = capacity;
			_top = 0;
		}
		~Stack()
		{
			cout << "~Stack()" << endl;
			free(_a);
			_capacity = _top = 0;
			_a = nullptr;
		}
			private:
		int* _a;
		int _capacity;
		int _top;
		};
int main() {
	Stack st1;
	Stack st2(st1);
	return 0;
}
运行后会变成
 
 从上面我们可以看出程序崩溃了,这是为什么呢?
 原因在于我们的栈的结构体类型中有一个指针来指向下一个结构体,当我们进行浅拷贝的时候会将这个地址也拷贝过去,但是我们的c++会自动调用析构函数,那么析构函数就被调用了两次,从上面的图中我们也可以看出来析构函数被调用了两次,所以程序崩溃了。
那么如何解决这个问题呢,我们的C++祖师爷,就定义了一个拷贝构造函数 来解决这个问题。
二、拷贝构造函数
1、概念
拷贝构造函数:只有单个形参,该形参是对本类类型对象的引用(一般常用const修饰),在用已存在的类类型对象创建新对象时由编译器自动调用。
2、特征
拷贝构造函数也是特殊的成员函数,其特征如下:
1. 拷贝构造函数是构造函数的一个重载形式。
2. 拷贝构造函数的参数
拷贝构造函数的参数只有一个且必须是类类型对象的引用,使用传值方式编译器直接报错,因为会引发无穷递归调用。
下面写一个拷贝构造函数重新进行运行:
#include<iostream>
using namespace std;
class Stack
	{
	public:
		Stack(size_t capacity = 3)
		{
			cout << "Stack(size_t capacity = 3)" << endl;
			_a = (int*)malloc(sizeof(int) * capacity);//要重新开辟空间
			if (nullptr == _a)
			{
				perror("malloc申请空间失败!!!");
			}
			_capacity = capacity;
			_top = 0;
		}
		//拷贝构造函数
		Stack(const Stack& stt ) {
			_a = (int*)malloc(sizeof(int) * stt._capacity);
			if (_a == nullptr) {
				perror("malloc");
				exit(-1);
			}
			memcpy(_a, stt._a, sizeof(int) * stt._top);
			_capacity = stt._capacity;
			_top = stt._top;
		}
		~Stack()
		{
			cout << "~Stack()" << endl;
			free(_a);
			_capacity = _top = 0;
			_a = nullptr;
		}
		private:
		int* _a;
		int _capacity;
		int _top;
		};
void Func(Stack stt) {
	//....
}
int main() {
	Stack st1;
	Func(st1);
	return 0;
}
就会发现是正常运行,如下
		//拷贝构造函数
		Stack(const Stack& stt ) {
			_a = (int*)malloc(sizeof(int) * stt._capacity);
			if (_a == nullptr) {
				perror("malloc");
				exit(-1);
			}
			memcpy(_a, stt._a, sizeof(int) * stt._top);
			_capacity = stt._capacity;
			_top = stt._top;
		}
通过拷贝构造函数我们可以看出,是重新开辟了一块空间进行拷贝构造,而且我们使用了引用(&)。那么为什么要用引用呢?
 下面我们用日期类函数进行演示:
#include<iostream>
using namespace std;
class Date {
public:
	Date(int year=1, int month=1, int day=1) {
		_year = year;
		_month = month;
		_day = day;
	}
	//错误的拷贝构造函数
	Date(Date d) {
		_year = d._year;
		_month = d._month;
		_day = d._day;
	}
	void Print() {
		cout << _year << "/" << _month << "/" << _day << endl;
	}
private:
	int _year ;
	int _month ;
	int _day ;
};
int main() {
	Date d1;
	d1.Print();
	Date d2(d1);
	d2.Print();
	return 0;
}
我们会发现代码没有办法运行,进行下面的报错
 
 这是由于当我们d2要对d1进行拷贝构造时发生了以下过程:
 规定传值传参都要去调用拷贝构造函数那么,中间就还有临时变量要创建和拷贝,这样一环套一环没有终点。如下:
 
 综上,所以我们要用引用,直接将d1赋值给d,如下所示:
 
#include<iostream>
using namespace std;
class Date {
public:
	Date(int year=1, int month=1, int day=1) {
		_year = year;
		_month = month;
		_day = day;
	}
	//正确的拷贝构造函数
	Date(Date& d) {
		_year = d._year;
		_month = d._month;
		_day = d._day;
	}
	void Print() {
		cout << _year << "/" << _month << "/" << _day << endl;
	}
private:
	int _year ;
	int _month ;
	int _day ;
};
int main() {
	Date d1;
	d1.Print();
	Date d2(d1);
	d2.Print();
	return 0;
}
3. 若未显式定义,编译器会生成默认的拷贝构造函数。
默认的拷贝构造函数对象按内存存储按字节序完成拷贝,这种拷贝叫做浅拷贝,或者值拷贝。
 注意:类中如果没有涉及资源申请时,拷贝构造函数是否写都可以;一旦涉及到资源申请时,则拷贝构造函数是一定要写的,否则就是浅拷贝。
所以其实日期类是不用进行我们自己写拷贝构造函数的,因为日期类浅拷贝就够用了,我在上面用日期类进行举例是为了方便我们理解。拷贝构造还是主要用在我开始写的栈类型的程序上。
#include<iostream>
using namespace std;
class Stack
	{
	public:
		Stack(size_t capacity = 3)
		{
			cout << "Stack(size_t capacity = 3)" << endl;
			_a = (int*)malloc(sizeof(int) * capacity);//要重新开辟空间
			if (nullptr == _a)
			{
				perror("malloc申请空间失败!!!");
			}
			_capacity = capacity;
			_top = 0;
		}
		//拷贝构造函数
		Stack(const Stack& stt ) {
			_a = (int*)malloc(sizeof(int) * stt._capacity);
			if (_a == nullptr) {
				perror("malloc");
				exit(-1);
			}
			memcpy(_a, stt._a, sizeof(int) * stt._top);
			_capacity = stt._capacity;
			_top = stt._top;
		}
		~Stack()
		{
			cout << "~Stack()" << endl;
			free(_a);
			_capacity = _top = 0;
			_a = nullptr;
		}
		private:
		int* _a;
		int _capacity;
		int _top;
		};
void Func(Stack stt) {
	//....
}
int main() {
	Stack st1;
	Func(st1);
	Stack st2(st1);
	return 0;
}
通过调试上述代码我们发现,它们_a的地址不同,但是_capacity、_top的值是相同的,成功完成了拷贝构造。
 
 运行后的结果如下:
 
4. 拷贝构造函数典型调用场景
使用已存在对象创建新对象
 函数参数类型为类类型对象
 函数返回值类型为类类型对象



















