前言

  前面我们学习完了设计模式,在其中我们有了解到原型模式。这里涉及到了克隆自身对象。那么也就是对对象进行拷贝。这里就涉及到了这么一个概念。深浅拷贝、何为深拷贝何为浅拷贝呢?我们一起来看看吧。

浅拷贝

  首先我们看看浅拷贝。浅拷贝就是将对象中的所有字段复制到新对象中去,浅拷贝对于值类型和引用类型有不同的影响。值类型的值被复制到副本中后,修改副本中的值不会影响原来对象的值。然而引用类型被复制到副本中的是引用类型的引用。不是引用的对象。这样再修改副本中的值是会导致原来对象的值也被修改了。但是这里引用类型情况我们需要排除字符串String类型。

  那么为何引用类型修改副本的值会造成原来对象的值的变化,而string字符串类型却排除在外呢?首先我们需要知道这么一个概念,string类型是一个不可变的数据类型,也就是意味着对字符串对象进行了初始化,该字符串对象就不能改变了。表面上我们修改字符串的内容的方法和运算实际上是创建了一个新字符串,然后根据需要可以把旧字符串的内容复制到新字符串中。怎么理解你?我们看下面这个案例:

        #region  字符串比较
/// <summary>
/// 获取引用类型的内存地址方法
/// </summary>
/// <param name="o"></param>
/// <returns></returns>
public static string getMemory(object o)
{
GCHandle h = GCHandle.Alloc(o, GCHandleType.Pinned);
IntPtr addr = h.AddrOfPinnedObject();
return "0x" + addr.ToString("X");
}
/// <summary>
/// 字符串比较
/// </summary>
public static void Compares()
{
string a = "";
Console.WriteLine("a的引用地址:\t\t" + getMemory(a));
string b = "";
Console.WriteLine("b的引用地址:\t\t" + getMemory(b));
Console.WriteLine("a与b的比较:\t\t" + Object.ReferenceEquals(a, b));
b = "";
Console.WriteLine("b的引用地址:\t\t" + getMemory(b)); } #endregion

  这里我们看a=”123”,b=”123”。我们看他们的引用地址是一样的。也就是说我们先创建a的时候创建了字符串a,有了一个引用地址。然后我们创建b的时候首先会寻找是否存在相同的值。如果存在相同的值就获取其引用地址。这也就是为什么a与b的引用地址是一样的。这里涉及到一个叫做字符驻留池的东西。会对字符串进行保存。那么后面我们修改b的值然后输出其引用地址,发现和之前的引用地址不一样。说明并不是修改原来的值,而是重新创建了一个字符串,重新获取了它的引用地址。

  我们接下来看一个浅拷贝的案例吧,首先我们准备的是以下的数据类型的值:int,string,enum,struct,class,int[],string[]。

    /// <summary>
/// 枚举
/// </summary>
public enum EnumTest
{
TestOne = ,
TestTwo =
} /// <summary>
/// 结构体
/// </summary>
public struct StructTest
{
public int Test;
public StructTest(int i)
{
Test = i;
}
} /// <summary>
/// 类
/// </summary>
public class ClassTest
{
public string TestString;
public ClassTest(string _string)
{
TestString = _string;
}
}
/// <summary>
/// 深拷贝
/// </summary>
public class DeepClone : ICloneable
{
public int _int = ;
public string _string = "";
public EnumTest _enum = EnumTest.TestOne;
public StructTest _struct = new StructTest();
public ClassTest _class = new ClassTest("");
public int[] arrInt = new int[] { };
public string[] arrString = new string[] { "" };
public object Clone()
{
var NewOne = JsonConvert.SerializeObject(this);
return JsonConvert.DeserializeObject<DeepClone>(NewOne);
}
}
class Program
{
static void Main(string[] args)
{
DeepClone simple = new DeepClone();
var simpleTwo = (DeepClone)simple.Clone();
simpleTwo._int = ;
simpleTwo._string = "";
simpleTwo._enum = EnumTest.TestTwo;
simpleTwo._struct.Test = ;
simpleTwo._class.TestString = "";
simpleTwo.arrInt[] = ;
simpleTwo.arrString[] = ""; Console.WriteLine($"int 类型变化 原对象:{simple._int}\t\t 备份对象:{simpleTwo._int}");
Console.WriteLine($"string 类型变化 原对象:{simple._string}\t\t 备份对象:{simpleTwo._string}");
Console.WriteLine($"enum 类型变化 原对象:{(int)simple._enum}\t\t 备份对象:{(int)simpleTwo._enum}");
Console.WriteLine($"struct 类型变化 原对象:{simple._struct.Test}\t\t 备份对象:{simpleTwo._struct.Test}");
Console.WriteLine($"class 类型变化 原对象:{simple._class.TestString}\t\t 备份对象:{simpleTwo._class.TestString}");
Console.WriteLine($"int数组 类型变化 原对象:{simple.arrInt[0]}\t\t 备份对象:{simpleTwo.arrInt[0]}");
Console.WriteLine($"string数组 类型变化 原对象:{simple.arrString[0]}\t\t 备份对象:{simpleTwo.arrString[0]}");
}
}

  我们通过继承ICloneable接口对这些类型都进行了浅拷贝然后修改副本对象。输出原对象和副本对象进行比较。我们发现int,enum,struct、值类型以及string这个特殊的引用类型的原对象值没有被影响改变。但是class,int[],string[]这些引用类型对象原对象被影响改变了值。也就再次验证了我们前面说的。浅拷贝是将对象进行赋值到一个副本对象中去,值类型复制值,引用类型复制其引用对象。修改副本对象值,值类型和string原对象不会被影响改变,引用类型除string其原对象都会被影响改变。

深拷贝

  我们上面看了浅拷贝,浅拷贝还是有一定的影响的,处理不好可能就成bug。那么我们看看对应的深拷贝又是什么样的呢?这里可以先声明,深拷贝对值类型和引用类型都没有区别对待。深拷贝也是将对象中的所有字段复制到新对象中去,但是对象无论是值类型还是引用类型都将被重新创建然后复制到副本对象去。对于副本对象的修改将不会影响到原对象,无论任何类型。

  我们继续将上面的例子进行深拷贝看看:

    /// <summary>
/// 深拷贝
/// </summary>
public class DeepClone : ICloneable
{
public int _int = ;
public string _string = "";
public EnumTest _enum = EnumTest.TestOne;
public StructTest _struct = new StructTest();
public ClassTest _class = new ClassTest("");
public int[] arrInt = new int[] { };
public string[] arrString = new string[] { "" };
public object Clone()
{
var NewOne = JsonConvert.SerializeObject(this);
return JsonConvert.DeserializeObject<DeepClone>(NewOne);
}
} class Program
{
static void Main(string[] args)
{
DeepClone simple = new DeepClone();
var simpleTwo = (DeepClone)simple.Clone();
simpleTwo._int = ;
simpleTwo._string = "";
simpleTwo._enum = EnumTest.TestTwo;
simpleTwo._struct.Test = ;
simpleTwo._class.TestString = "";
simpleTwo.arrInt[] = ;
simpleTwo.arrString[] = ""; Console.WriteLine($"int 类型变化 原对象:{simple._int}\t\t 备份对象:{simpleTwo._int}");
Console.WriteLine($"string 类型变化 原对象:{simple._string}\t\t 备份对象:{simpleTwo._string}");
Console.WriteLine($"enum 类型变化 原对象:{(int)simple._enum}\t\t 备份对象:{(int)simpleTwo._enum}");
Console.WriteLine($"struct 类型变化 原对象:{simple._struct.Test}\t\t 备份对象:{simpleTwo._struct.Test}");
Console.WriteLine($"class 类型变化 原对象:{simple._class.TestString}\t\t 备份对象:{simpleTwo._class.TestString}");
Console.WriteLine($"int数组 类型变化 原对象:{simple.arrInt[0]}\t\t 备份对象:{simpleTwo.arrInt[0]}");
Console.WriteLine($"string数组 类型变化 原对象:{simple.arrString[0]}\t\t 备份对象:{simpleTwo.arrString[0]}");
}
}

这里我们看这个运行结果,无论值类型还是引用类型修改副本对象之后都没有影响原对象的值。这也就是深拷贝的特点了。

总结

  我们看完了浅拷贝与深拷贝,我们仔细回顾下。浅拷贝将对象的字段复制到新的对象中去,但是当修改新对象的时候,值类型和string类型的字段将不会影响原对象的字段,而引用类型除string类型外都将影响原对象的值。深拷贝也是将对象的字段复制到新的对象中去,但是无论是值类型还是引用类型的改变都不会影响原对象的值。因为深拷贝是将原对象重新创建然后复制到副本对象中去的。


  人生只有走出来的美丽,没有等出来的辉煌。

    欢迎大家扫描下方二维码,和我一起学习更多的知识

C#解析深浅拷贝的更多相关文章

  1. Python基础【3】:Python中的深浅拷贝解析

    深浅拷贝 在研究Python的深浅拷贝区别前需要先弄清楚以下的一些基础概念: 变量--引用--对象(可变对象,不可变对象) 切片(序列化对象)--拷贝(深拷贝,浅拷贝) 我是铺垫~ 一.[变量--引用 ...

  2. 关于:1.指针与对象;2.深浅拷贝(复制);3.可变与不可变对象;4.copy与mutableCopy的一些理解

    最近对深浅拷贝(复制)做了一些研究,在此将自己的理解写下来,希望对大家有所帮助.本人尚处在摸索阶段,希望各位予以指正. 本文包括如下方向的探索: 1.指针与对象: 2.深/浅拷贝(复制): 3.可变/ ...

  3. python基础知识9---字符串拼接,深浅拷贝,三元运算

    一.字符串格式化 Python的字符串格式化有两种方式: 百分号方式.format方式 百分号的方式相对来说比较老,而format方式则是比较先进的方式,企图替换古老的方式,目前两者并存.[PEP-3 ...

  4. day07 深浅拷贝

    今日概要 深浅拷贝(重点) 文件操作 详细内容 直接赋值: 直接将对象的引用赋值给另一个对象 v1=1000 v2=v1 #v1 v2指向同一个内存地址 print(id(v1),id(v2))#相等 ...

  5. python运算符,数据类型,数据类型操作,三目运算,深浅拷贝

    算数运算符: Py2中精确除法需要导入:from __future__ import division,(符由特  ,将来的.滴未省,除法) py3不需要导入 赋值运算符: 比较运算符: 成员运算符: ...

  6. 关于Java中的HashMap的深浅拷贝的测试与几点思考

    0.前言 工作忙起来后,许久不看算法,竟然DFA敏感词算法都要看好一阵才能理解...真是和三阶魔方还原手法一样,田园将芜,非常可惜啊. 在DFA算法中,第一步是需要理解它的数据结构,在此基础上,涉及到 ...

  7. 人生苦短之我用Python篇(深浅拷贝、常用模块、内置函数)

    深浅拷贝 有时候,尤其是当你在处理可变对象时,你可能想要复制一个对象,然后对其做出一些改变而不希望影响原来的对象.这就是Python的copy所发挥作用的地方. 定义了当对你的类的实例调用copy.c ...

  8. python之基本数据类型及深浅拷贝

    一.数据基本类型之set集合 set和dict类似,也是一组key的集合,但不存储value.由于key不能重复,所以,在set中,没有重复的key set集合,是一个无序且不重复的元素集合 1.创建 ...

  9. Python全栈开发之3、深浅拷贝、变量和函数、递归、函数式编程、内置函数

    一.深浅拷贝 1.数字和字符串 对于 数字 和 字符串 而言,赋值.浅拷贝和深拷贝无意义,因为其永远指向同一个内存地址. import copy # 定义变量 数字.字符串 # n1 = 123 n1 ...

随机推荐

  1. 判断java中最多的词组

    其中的难点,是空格,以及如果第一个是空格怎么办,虽然事后看着很简单,但是做的时候却十分的困难! static void Daunyu()throws IOException {     Word wo ...

  2. Tomcat+Nginx+Linux+Mysql部署豆瓣TOP250的项目到腾讯云服务器

    写在前面 因为前面有写过一篇关于豆瓣的top250的电影的可视化展示项目,你可以移步http://blog.csdn.net/liuge36/article/details/78607955了解这个项 ...

  3. 阿里云服务器CentOS6.9安装Mysql

    上篇讲了CentOS6.9安装tomcat,这篇来讲mysql的安装 1.查看CentOS是否安装了MySQL yum list installed | grep mysql //查看CentOS是否 ...

  4. Kubernetes 系列(四):使用Traefik访问.net core api

    一. 准备 本篇的要求是在前三篇的基础上已经搭建好的本地k8s以及部署了Traefik,我们将会使用Traefik Ingress来访问.net core api,比较简单,做个记录,如果还没有搭建k ...

  5. vue-hash-calendar,移动端日期时间选择插件

    按照惯例,先上效果图 vue-hash-calendar 基于 vue 2.X 开发的日历组件 支持手势滑动操作·1 原生 js 开发,没引入第三方库 上下滑动 切换 周/月 模式 [周模式中] 左右 ...

  6. KafkaStream简介

    Kafka Streams 1 概述 Kafka Streams是一个客户端程序库,用于处理和分析存储在Kafka中的数据,并将得到的数据写回Kafka或发送到外部系统.Kafka Stream基于一 ...

  7. STM32之串口DMA接收不定长数据

    STM32之串口DMA接收不定长数据 引言 在使用stm32或者其他单片机的时候,会经常使用到串口通讯,那么如何有效地接收数据呢?假如这段数据是不定长的有如何高效接收呢? 同学A:数据来了就会进入串口 ...

  8. C语言--最大公约数

    //辗转相除法 int main() { int a,b; int t; scanf("%d %d", &a,&b); ) { t = a%b; a = b; b ...

  9. VMware workstation Windows 10虚拟机安装步骤

    1. 在首页点击创建新的虚拟机 2. 选择典型,下一步 3.选择稍后安装操做系统,下一步. 4. 选择第一个Microsoft Windows,版本选择如图. 5. 选择安装位置,我放在D盘. 7. ...

  10. IDA快捷键整理

    空格键 反汇编窗口切换文本跟图形 ESC退到上一个操作地址 G搜索地址或者符号 N重命名 分号键 注释 ALT+M 添加标签 CTRL+M 列出所有标签 CTRL +S 二进制段的开始地址结束地址 C ...