拷贝构造函数
- Copy构造函数的定义
拷贝构造函数,是一种特殊的构造函数,它由编译器调用来完成一些基于同一类的其他对象的构建及初始化。其唯一的参数(对象的引用)是不可变的(const类型)。
- 什么情形下调用拷贝构造函数
- 一个对象以值传递的方式传入函数体;
- 一个对象以值传递的方式从函数返回;
- 一个对象需要通过另外一个对象进行初始化;
- 如果在前两种情况不使用拷贝构造函数的时候,就会导致一个指针指向已经被删除的内存空间。对于第三种情况来说,初始化和赋值的不同含义是构造函数调用的原因。事实上,拷贝构造函数是由普通构造函数和赋值操作符共同实现的。
- 拷贝构造函数不可以改变它所引用的对象。
- 如果在类中没有显式的声明一个拷贝构造函数,那么,编译器会自动生成一个来进行对象之间的位拷贝(Bitwise Copy)。
- 拷贝构造函数的优势
- 因为它不用在构造一个对象的时候改变构造函数的参数列表。
- 设计拷贝构造函数是一个良好的风格,即使是编译系统会自动为你生成默认拷贝构造函数。
- 事实上,默认拷贝构造函数可以应付许多情况。
示例
下面一个代码就是调用了拷贝构造函数
#include <iostream>
#include <cstring>
using namespace std;
class Person
{
char *name; //姓名
int age; //年龄
public:
Person(); //无参的构造函数
Person(char *na, int a = 18); //带有默认参数值的构造函数
Person(const Person &p);
~Person();
void disp();
};
Person::Person() //无参的构造函数
{
cout << "调用 Person::Person():" << endl;
name = NULL;
age = 0;
}
Person::Person(char *na, int a)
{
cout << "调用 Person::Person(char *na,int a):" << endl;
int len = strlen(na) + 1; //确定要申请的内存空间的大小
if (name != NULL)
{
delete name;
}
name = new char[len]; //为数据成员name申请内存空间
strcpy_s(name, len, na);
age = a; //用形参变量a的值初始化对象的age成员
}
Person::Person(const Person&p)
{
cout << "调用 Person::Person(const Person&p):" << endl;
int len = strlen(p.name) + 1;
if (name != NULL)
{
delete name;
name = NULL;
}
name = new char[len];
strcpy_s(name, len, p.name);
age = p.age;
}
Person::~Person()
{
if (name != NULL)
delete name;
name = NULL;
}
void Person::disp()
{
if (name != NULL)
cout << name << "\t";
cout << age << endl;
}
int main()
{
Person p1; //调用不带参的构造函数
Person p2("Hellen", 20); //调用带参的构造函数
Person p3 = { "Bjarne" }; //调用带默认参数值的构造函数,C++11支持,列表初始化语法
p1.disp();
p2.disp();
p3.disp();
Person p4(p3);
p4.disp();
fflush(stdin); getchar();
return 0;
}
运行结果
调用 Person::Person():
调用 Person::Person(char *na,int a):
调用 Person::Person(char *na,int a):
0
Hellen 20
Bjarne 18
调用 Person::Person(const Person&p):
Bjarne 18