C++深拷贝

对于一般的对象比如
int a = 10;
int b = 20;
对象间的赋值,复制过程很简单,但对于类对象来说,其内部存在着各种各样类型变量,其拷贝过程比较复杂。事实上,在对象拷贝过程中。如果没有自定义拷贝构造函数,系统会提供一个默认的拷贝构造函数,缺省的拷贝构造函数对于基本类型的成员变量,按字节复制,对于类类型的成员变量,调用其相应得拷贝构造函数,原型如下:

string(const string& that){}

这对于一般的类型变量拷贝没有问题,但如果类中含有指针类型的变量使用系统的缺省拷贝构造函数,就会出现问题,因为缺省的拷贝构造函数是按字节复制的,对于指针类型的成员变量只复制指针本身,而不复制指针向的对象这被称为浅拷贝,浅拷贝会带来指针被重复释放的问题,在运行阶段会出现断错误。比如如下代码

#include<iostream>
#include<cstring>
using namespace std;
class String {
public:
	String (const char* psz=NULL) : m_psz(strcpy(new char[strlen(psz?psz:"")+1],(psz?psz:""))){
		cout << "String构造" << endl;
	}
	~String () {
		if(m_psz) {
			delete[] m_psz;
			m_psz = NULL;
		}
		cout << "String析构" << endl;
	}
	char* c_str(void) {
		return m_psz;
	}
private:
	char* m_psz;
};
int main(void) {
	String s1("hello");
	String s2(s1);
	cout << "s1    " << s1.c_str() << endl;
	cout << "s2    " << s2.c_str() << endl;
	s1.c_str()[0] = 'H';
	cout << "s1    " << s1.c_str() << endl;
	cout << "s2    " << s2.c_str() << endl;
	return 0;
}
上述代码中,s1,s2对象中的m_psz指针指向同一块内存区域"hello",所以将s1中的m_psz指向的内存区域的第一个字符改为大写的"H",s2输出也是大写“H”,而且更要命的是当对象释放的时候,该内存会被释放两次导致程序崩溃。
在程序中加入我们自己定义的拷贝构造函数,如下

String (const String& that) : m_psz(strcpy((new char[strlen(that.m_psz)+1]),that.m_psz)){
cout << "String拷贝构造" << endl;
}

这样再运行就没有问题了。
当函数存在对象型的参数或对象型的返回值时都会用到拷贝构造函数。而拷贝赋值的情况基本上与拷贝复制是一样的。只是拷贝赋值是属于操作符重载问题。我们知道对于普通变量来讲a=b返回的是左值a的引用,所以它可以作为左值继续接收其他值(a=b)=30,这样来讲我们操作符重载后返回的应该是类对象的引用(否则返回值将不能作为左值来进行运算)

String& operator=(cosnt String& that) {
    if(&that != this) {//防止自赋值
        char *psz = strcpy (new char[strlen(that.m_psz)+1],that.m_psz);//如果失败会抛出异常,m_psz最后在析构函数里释放
        delete[] m_psz;
        m_psz = psz;   
    }
    return *this;
}


不知道怎么多了下面这些代码格式






  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
设计并实现一个动态整型数组类Vect,要求: (1)实现构造函数重载,可以根据指定的元素个数动态创建初始值为0的整型数组,或根据指定的内置整型数组动态创建整型数组。 (2)设计拷贝构造函数和析构函数,注意使用深拷贝。 (3)设计存取指定位置的数组元素的公有成员函数,并进行下标越界,若越界则输出“out of boundary”。 (4)设计获取数组元素个数的公有成员函数。 (5)设计用于输出数组元素的公有成员函数,元素之间以空格分隔,最后以换行符结束。 在main函数中按以下顺序操作: (1)根据内置的静态整型数组{1,2,3,4,5}构造数组对象v1,根据输入的整型数构造数组对象v2。 (2)调用Vect的成员函数依次输出v1和v2的所有元素。 (3)输入指定的下标及对应的整型数,设置数组对象v1的指定元素。 (4)根据数组对象v1拷贝构造数组对象v3。 (5)调用Vect的成员函数依次输出v1和v3的所有元素。 设计并实现一个动态整型数组类Vect,要求: (1)实现构造函数重载,可以根据指定的元素个数动态创建初始值为0的整型数组,或根据指定的内置整型数组动态创建整型数组。 (2)设计拷贝构造函数和析构函数,注意使用深拷贝。 (3)设计存取指定位置的数组元素的公有成员函数,并进行下标越界,若越界则输出“out of boundary”。 (4)设计获取数组元素个数的公有成员函数。 (5)设计用于输出数组元素的公有成员函数,元素之间以空格分隔,最后以换行符结束。 在main函数中按以下顺序操作: (1)根据内置的静态整型数组{1,2,3,4,5}构造数组对象v1,根据输入的整型数构造数组对象v2。 (2)调用Vect的成员函数依次输出v1和v2的所有元素。 (3)输入指定的下标及对应的整型数,设置数组对象v1的指定元素。 (4)根据数组对象v1拷贝构造数组对象v3。 (5)调用Vect的成员函数依次输出v1和v3的所有元素。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值