【C++】深拷贝和浅拷贝

目录

浅拷贝

 深拷贝

字符串的构造

例有两个类的深拷贝:


浅拷贝

在类中,若我们不写拷贝构造函数,则程序会提供一个默认的拷贝构造函数,该函数为浅拷贝

   //默认拷贝构造 ---浅拷贝--值复制

类名(const 类名& 形参名)   {成员1= 形参名.成员1; }

class A
{
public:
	A(int i = 0, int j = 0) :m_i(i), m_j(j) {}//构造函数
	void print() { cout << m_i << "," << m_j << endl; }
private:
	int m_i;
	int m_j;
};
void main()
{
	A a(4, 7);
	A b(a);//使用默认拷贝构造函数
	a.print();//4,7
	b.print();//4,7
}

 

浅拷贝将一个对象的所有信息直接拷贝到另一个对象中,如果是值则拷贝值;如果是引用、指针则拷贝其内存地址,这时,两个对象的该成员共用一个地址。

由于对象最后都需要析构,若浅拷贝的对象中有指针,该地址在析构的时候将会进行两次(拷贝和被拷贝的对象每个一次),导致程序出错。

class A
{
public:
	A(int i = 0)
	{
		//m_i = new int(i);
		m_i = new int;
		*m_i = i;
	}
	~A()
	{
		if (m_i != NULL)
		{
			delete m_i;
			m_i = NULL;
		}

	}
	A(const A& a)  //默认拷贝构造 ---浅拷贝--值复制
	{
		m_i = a.m_i;  //两个对象的指针指向同一块内存单元
	}
	void print()
	{
		cout << *m_i << endl;
	}
private:
	int* m_i;
};
void main()
{
	A a(10);
	a.print();
	A b(a);  //调用默认拷贝构造,会导致b对象中的m_i和a对象中的m_i指向同一块内存空间
	b.print();
	//在即将退出时,调用析构会出问题,析构了两次
}

 深拷贝

如果有指针作为数据成员,则必须要写析构函数拷贝构造函数,拷贝资源。

深拷贝中如果对象有申请动态内存,则深拷贝中也会新申请一个(属性拷贝),析构函数中释放空间。

class A
{
public:
	A(int i = 0)
	{
		m_i = new int;
		*m_i = i;
	}
	A(const A& a)  //深拷贝
	{
		m_i = new int;//开新空间
		*m_i = *(a.m_i);
	}
	~A()
	{
		delete m_i;
		m_i = NULL;
	}
	void print() { cout << *m_i << endl; }
private:
	int* m_i;
};
void main()
{
	A a(20);
	A b(a);
	a.print();//20
	b.print();//20
}

字符串的构造

strcpy拷贝所有字符串
strncpy拷贝n个字符串
memcpy  void*memcpy(void*dest,void*src,count)都可以拷贝

class STR
{
public:
	STR(const char* str)
	{
		m_str = new char[strlen(str) + 1];
		strcpy_s(m_str, strlen(str) + 1, str);
	}
	STR()//什么都不传入,拿一个\0占位
	{
		m_str = new char[1];
		m_str[0] = '\0';
	}
	STR(int n, char c)
	{
		m_str = new char[n + 1];
		memset(m_str, c, n);//strcpy只能拷贝字符串,memset什么都可以拷贝
		m_str[n] = '\0';
	}
	STR(const STR& s)//深拷贝
	{
		m_str = new char[strlen(s.m_str) + 1];
		strcpy_s(m_str, strlen(s.m_str) + 1, s.m_str);
	}
	~STR()//析构
	{
		if (m_str != NULL)
		{
			delete[]m_str;
			m_str = NULL;
		}
	}
	void print()
	{
		cout << m_str << endl;
	}
private:
	char* m_str;
};
void main()
{
	STR s1("12345");
	STR s2;
	STR s3(4, 'k');
	STR s4(s1);
	s1.print();
	s2.print();
	s3.print();
	s4.print();
}

如果成员是指针可以new大小(上方代码),但数组不能,只能依靠数组定大小截断(如下)

就是说指针需要先申请空间(申请需要放入的大小)再往里面放东西,而数组因为本身有大小所以直接放入数组能存的大小的内容。

class STR
{
public:
	STR(const char* str)
	{
		int n = sizeof(m_str);
		memcpy(m_str, str, n - 1);
		m_str[n - 1] = '\0';
	}
	void print()
	{
		cout << m_str << endl;
	}
private:
	char m_str[20];
};
void main()
{
	STR a("asdf");
	STR b("asdfasdfasdfasddfasdfasdfadfgsdfgsdfgsdfg");
	a.print();
	b.print();
}

例有两个类的深拷贝:

class Date
{
private:
	int m_year;
	int m_month;
	int m_day;
public:
	Date(int y, int m, int d) :m_year(y), m_month(y), m_day(d) {}
	void show()
	{
		cout << m_year << " /" << m_month << "/" << m_day << endl;
	}
	Date(const Date& d) :m_year(d.m_year), m_month(d.m_month), m_day(d.m_day)
	{
		cout << "Data(&)" << endl;
	}
};
class Student
{
private:
	int m_num;
	char* m_name;
	char m_sex;
	Date m_birthday;
public:
	Student(int num, const char* name, char sex, int y, int m, int d) :m_num(num), m_sex(sex), m_birthday(y, m, d)//没有默认值必须显示的调用构造函数
	{
		m_name = new char[strlen(name) + 1];
		strcpy_s(m_name, strlen(name) + 1, name);
	}
	void print()
	{
		cout << m_num << " " << m_name << " " << m_sex;
		m_birthday.show();
	}
	~Student()
	{
		if (m_name != NULL)
		{
			delete[]m_name;
			m_name = NULL;
		}
	}
	Student(const Student& s) :m_num(s.m_num), m_sex(s.m_sex), m_birthday(s.m_birthday) //调用Date类的拷贝构造
	{
		m_name = new char[strlen(s.m_name) + 1];
		strcpy_s(m_name, strlen(s.m_name) + 1, s.m_name);
	}
};

void main()
{
	Student s1(1001, "wangpangpang", 'm', 2000, 12, 24);
	Student s2(s1);
	s2.print();
	s1.print();
}
  • 4
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

曦樂~

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值