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;
}

C++深浅拷贝和写时拷贝图文详解

而我们之前所说的浅拷贝崩溃是因为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;
}

C++深浅拷贝和写时拷贝图文详解

可以看出,三个对象的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");
}

C++深浅拷贝和写时拷贝图文详解

拷贝构造:s1和s2管理同一块空间

C++深浅拷贝和写时拷贝图文详解

定义s3,和s1、s2没有关联;

C++深浅拷贝和写时拷贝图文详解

我们再完全理一遍:

C++深浅拷贝和写时拷贝图文详解

此时已经解决我们之前提到过的第二个问题。

再来看第一个问题:

C++深浅拷贝和写时拷贝图文详解

s1的改变影响了s2;

写时拷贝:需要改变的时候深拷贝。

C++深浅拷贝和写时拷贝图文详解


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++;
}

C++深浅拷贝和写时拷贝图文详解

总结

来源:https://blog.csdn.net/qq_43560037/article/details/115425535

标签:c++,深浅拷贝,写时拷贝
0
投稿

猜你喜欢

  • 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
  • c# 服务器上传木马监控代码(包含可疑文件)

    2022-07-08 10:02:45
  • 关于Java中@SuppressWarnings的正确使用方法

    2023-07-22 17:17:58
  • C#修改及重置电脑密码DirectoryEntry实现方法

    2021-12-03 05:08:16
  • java面向对象:API(接口)与集合(ArrayList)

    2021-06-07 03:28:19
  • 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
  • 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
  • C# DataTable中查询指定字段名称的数据

    2023-10-08 16:16:44
  • MyBatis 动态拼接Sql字符串的问题

    2021-08-09 05:23:46
  • Java 3种方法实现进制转换

    2022-03-24 00:02:03
  • asp之家 软件编程 m.aspxhome.com