C++ 拷贝(复制)构造函数

拷贝构造函数用以将一个类的对象拷贝给同一个类的另一个对象,比如之前学习过的string类:

string s1;
string s2 = s1;

一般情况下的拷贝构造函数:

class A
{
private:
	int n;
	double d;
	char s;
public:
	A(const A& a);
};

A::A(const A& a)
{
	this->n = a.n;
	this->d = a.d;
	this->s = a.s;
}

即按照数据类型开辟一段内存空间用以存放拷贝进来的对象的数据。需要注意的是必须传递进来的是类的引用,如果是按值传递,将会生成一个临时的类的对象a,并将传递进来对象拷贝给临时对象,其实就是又调用了拷贝构造函数。

默认拷贝构造函数:

如果用户没有自定义拷贝构造函数,又使用了对象的拷贝,则编译器会自动生成一个默认构造函数,格式同上。

浅拷贝和深拷贝:

大多数情况下,使用默认构造函数就可以完成对象的拷贝(浅拷贝),但是当对象中有诸如指针、动态数组等数据类型时使用默认构造函数则可能出错,此时需要用户自定义拷贝构造函数(深拷贝),以下是一个例子,首先是没有自定义拷贝构造函数的情况:

class A
{
private:
	char* str;
	int len;
public:
	A(const char* s);
	~A();
	//A(const A& a);
};
A::A(const char* s)
{
	len = strlen(s);
	str = new char[len+1];
	strcpy(str, s);
	cout << str << " object construct." << endl;
}
A::~A()
{
	cout << str << " deleted." << endl;
	delete[]str;
}

//A::A(const A& a)
//{
//	this->len = a.len;
//	this->str = new char[a.len+1];
//	strcpy(str, a.str);
//}

调用函数:

int main(void)
{
	A a1("Hello");
	A a2 = a1;
	return 0;
}

运行结果:

Hello object construct.
Hello deleted.
葺葺葺葺葺葺葺葺攐? deleted.

这是因为在对象复制的时候,由于编译器生成了默认拷贝构造函数,只是单纯的将a1中指针str的值赋值给a2中的指针str,导致a2的生命周期结束时调用析构函数将str指向的内存空间内容释放掉了,于是a1生命周期结束时调用析构函数释放掉的内存中的内容就是无意义的字符了。

去掉注释后的正确写法:

class A
{
private:
	char* str;
	int len;
public:
	A(const char* s);
	~A();
	A(const A& a);
};
A::A(const char* s)
{
	len = strlen(s);
	str = new char[len+1];
	strcpy(str, s);
	cout << str << " object construct." << endl;
}
A::~A()
{
	cout << str << " deleted." << endl;
	delete[]str;
}

A::A(const A& a)
{
	this->len = a.len;
	this->str = new char[a.len+1];
	strcpy(str, a.str);
}

调用函数同上。

运行结果:

Hello object construct.
Hello deleted.
Hello deleted.

这里自定义了拷贝构造函数,申请了一块新的内存空间来存放拷贝进来的字符串,因此释放时就不会出错了。

参考:

C++拷贝构造函数(复制构造函数)详解

C++深拷贝和浅拷贝(深复制和浅复制)完全攻略

《深度探索C++对象模型 侯捷 译》

  • 11
    点赞
  • 44
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值