C++深浅拷贝和写时拷贝图文详解
作者:_Camille 时间:2021-06-07 00:12:58
前言
之前我们在浅谈6个成员函数中有提到深浅拷贝的问题,现在再回首掏一把。
一、深浅拷贝哪家强?
先给出代码理一理
#define _CRT_SECURE_NO_WARNINGS 1
#include <iostream>
#include<assert.h>
using namespace std;
class String
{
friend ostream& operator<<(ostream &out, const String &s);
public:
String(const char* str = "")
{
m_data = new char[strlen(str) + 1];
strcpy(m_data, str);
}
//String(const String& s)//qian拷贝
//{
//m_data = s.m_data;
//}
String(const String& s)//深拷贝
{
m_data = new char[strlen(s.m_data) + 1];
strcpy(m_data, s.m_data);
}
String& operator=(const String& s)
{
if (this != &s)
{
delete[]m_data;
m_data = new char[strlen(s.m_data) + 1];
strcpy(m_data, s.m_data);
}
return *this;
}
~String()
{
delete[]m_data;
m_data = nullptr;
}
private:
char* m_data;
};
ostream& operator<<(ostream &out, const String &s)
{
out << s.m_data;
return out;
}
void main()
{
String s1("abc");
String s2 = s1;
cout << "s1 = " << s1 << endl;
cout << "s2 = " << s2 << endl;
}
而我们之前所说的浅拷贝崩溃是因为doublefree的问题,因此我们可以定义一个引用计数器,来记录当前使用该值的对象数,如果数目大于1,则不释放内存。
class String
{
friend ostream& operator<<(ostream &out, const String &s);
public:
String(const char* str = "")
{
m_data = new char[strlen(str) + 1];
strcpy(m_data, str);
m_count++;
}
String(const String& s)//浅拷贝
{
m_data = s.m_data;
m_count++;
}
String& operator=(const String& s)
{
if (this != &s)
{
m_data = s.m_data;
m_count++;
}
return *this;
}
~String()//浅赋值
{
if (--m_count == 0)
{
delete[]m_data;
m_data = nullptr;
}
}
private:
char* m_data;
static int m_count;//引用计数器
};
int String::m_count = 0;
ostream& operator<<(ostream &out, const String &s)
{
out << s.m_data;
return out;
}
void main()
{
String s1("abc");
String s2 = s1;
String s3;
s3 = s2;
cout << "s1 = " << s1 << endl;
cout << "s2 = " << s2 << endl;
cout << "s3 = " << s3 << endl;
}
可以看出,三个对象的m_data共享同一块内存空间,节省了资源;
但是暴露出了很多的问题:站在对象的角度,其中一个对象改变m_data其他的对象也会随之改变;其二若s3使用其他字符串初始化,但计数器还是三者共享。
倘若我们使用深拷贝方法,就不会出现这种问题。如果可以在不改变m_data前使用浅拷贝,在改变时使用深拷贝,暨同时实现深浅拷贝,那么就两全其美。
二、写时拷贝
通过对上面问题的分析,我们需要实现:引用计数器管理不同的空间。
class String_rep
{
public:
String_rep(const char* str = "") :m_count(0)
{
m_data = new char[strlen(str) + 1];
strcpy(m_data, str);
cout << "creat" << endl;
}
String_rep(const String_rep &rep) :m_count(0)
{
m_data = rep.m_data;
increment();
}
String_rep & operator=(const String_rep &rep)
{
if (this != &rep)
{
m_data = rep.m_data;
increment();
}
return *this;
}
public:
void increment()
{m_count++;}
void decrement()
{m_count--;}
private:
char* m_data;
int m_count;
};
class String
{
public:
String(const char* str = "") :pn(new String_rep(str))
{
pn->increment();
}
~String()
{
cout << "Free" << endl;
}
private:
String_rep *pn;
};
void main()
{
String s1("abc");
}
拷贝构造:s1和s2管理同一块空间
定义s3,和s1、s2没有关联;
我们再完全理一遍:
此时已经解决我们之前提到过的第二个问题。
再来看第一个问题:
s1的改变影响了s2;
写时拷贝:需要改变的时候深拷贝。
void to_upper()
{
String_rep *new_pn = new String_rep(pn->Getdata());//创建新空间
pn->decrement();//原空间计数器减一
pn = new_pn;//需要更改的对象的pn指向新空间
pn->increment();//新空间的计数器加一
char* p = pn->Getdata();
while (*p != '\0')
{
if (*p >= 'a' && *p <= 'z')
*p -= 32;
p++;
}
总结
来源:https://blog.csdn.net/qq_43560037/article/details/115425535
标签:c++,深浅拷贝,写时拷贝
![](/images/zang.png)
![](/images/jiucuo.png)
猜你喜欢
Hibernate的一对一,一对多/多对一关联保存的实现
2022-03-13 12:41:08
基于Elasticsearch5.4的常见问题总结
2022-07-15 15:02:49
Java中Steam流的用法详解
2021-12-16 14:18:50
springboot相关面试题汇总详解
2023-10-06 17:16:11
java二维数组指定不同长度实例方法
2021-07-13 06:02:09
Java高版本Api在Android中的使用方法详解
2023-01-19 20:03:08
![](https://img.aspxhome.com/file/2023/3/104023_0s.jpg)
c# 服务器上传木马监控代码(包含可疑文件)
2022-07-08 10:02:45
关于Java中@SuppressWarnings的正确使用方法
2023-07-22 17:17:58
![](https://img.aspxhome.com/file/2023/9/77109_0s.jpg)
C#修改及重置电脑密码DirectoryEntry实现方法
2021-12-03 05:08:16
java面向对象:API(接口)与集合(ArrayList)
2021-06-07 03:28:19
![](https://img.aspxhome.com/file/2023/0/125590_0s.png)
SpringBoot Java后端实现okhttp3超时设置的方法实例
2022-11-06 04:56:03
C# listview 点击列头排序的实例
2023-11-26 20:09:05
spring boot 使用profile来分区配置的操作
2022-11-27 22:55:15
![](https://img.aspxhome.com/file/2023/5/64205_0s.png)
Springboot整合mqtt服务的示例代码
2022-07-20 02:58:01
MyBatis在注解上使用动态SQL方式(@select使用if)
2023-09-24 06:52:55
C++编程中的const关键字常见用法总结
2023-12-02 17:41:48
Dubbo实现分布式日志链路追踪
2023-08-23 21:00:54
![](https://img.aspxhome.com/file/2023/2/58292_0s.png)
C# DataTable中查询指定字段名称的数据
2023-10-08 16:16:44
MyBatis 动态拼接Sql字符串的问题
2021-08-09 05:23:46
Java 3种方法实现进制转换
2022-03-24 00:02:03