C#深度复制和浅度复制

复制一个值变量很简单,新建一个变量然后将原来的变量赋值过去就行,但是复制一个引用变量这种方法是不行的,如果不明白为什么可以先看看这篇解释

引用类型变量和值类型变量在赋值时的不同

如果要复制一个引用类型的变量,比如说类,需要在类定义中继承ICloneable接口,并实现Clone方法,这是一个固定格式,下面看一个例子

 public class Test:ICloneable
{
public int Val;
public object Clone() => MemberwiseClone();
}

定义了一个Test类,继承ICloneable接口,实现Clone方法,实现Clone方法的格式是固定的,这里使用了简化写法,public object Clone() => MemberwiseClone();等于

public object Clone()
{
return MemberwiseClone();
}

这是C#提供的方法,按照这么写就对了,此时如果要复制一个Test类的引用变量,就可以这么写

Test t1 = new Test();
Test t2 = (Test)t1.Clone();

有一个引用类型变量t1,调用t1.Clone()会在堆中重新开辟一个内存空间,并且复制t1堆空间的值,然后会返回这个新空间的内存地址,因为Clone()方法返回类型是object,所以强制类型转换为Test,然后赋给Test类型的变量t2

这个过程,便是C#中的浅度复制(ShallowCopy),也有称为影子复制的

这个复制会存在一些问题,那就是一个引用类型中存在引用类型字段时,引用类型字段并不会也复制一份

public class Test:ICloneable
{
public int Val;
public Test2 Z = new Test2();
public object Clone()
{
return MemberwiseClone();
}
}
public class Test2
{
public int D;
}

这里定义了两个类,其中Test类中包含了一个Test2类型的引用类型变量Z,我们先看内存中怎样表示的

从这张图我们可以看到,当我们使用Clone对引用类型进行Clone时,只会复制堆空间的值,如果堆空间中有引用类型,在复制时就只单纯的复制了引用类型的堆空间地址,这样的后果就是虽然Clone得到了两个堆空间对象,但是堆空间对象中的字段却指向了同一个另外的空间地址,在某些情况下就会出现问题

所以

浅度复制:解决了直接使用赋值运算符时变量指向了同一个堆空间的问题(即Test t2 = t1的问题),但是只解决了一层,对于包含在内的堆空间中的成员的引用没有进一步解决(即t1.Z和t2.Z指向了同一个堆空间地址)

为了解决上述问题,就需要使用深度复制,深度复制的基本思路,就是在Clone方法中创建一个新的对象并使其内容与现有内容保持一致(其实是废话),我们甚至可以不用继承ICloneable,但是建议继承并使用Clone方法,这样可以保持方法名的一致性

至于怎么实现,可以自行思考(主要是我也在思考中)

所以这篇文章的主要内容是讲:C#提供的return MemberwiseClone();只是浅度复制,不会为引用成员创建新空间并将引用成员空间的值复制过去,只会复制引用成员的空间地址,需要注意

C#深度复制和浅度复制的更多相关文章

  1. .NET基础之深度复制和浅度复制

    之前一直没有搞清楚深度复制和浅度复制的区别到底在哪里,今天彻底把这个东西弄懂了,写出来与到家共勉. 如果大家不懂值类型和引用类型的区别,请先看http://www.cnblogs.com/Autumo ...

  2. c#学习笔记-深度复制 与浅度复制

    关于值类型和引用类型: 浅度复制(shallow copy)只复制值类型(char,int )的值,而对于引用类型不会复制,浅度复制可以通过派生于System.Object的MemberwiseClo ...

  3. 转载---Java集合对象的深度复制与普通复制

    原博文:http://blog.csdn.net/qq_29329775/article/details/49516247 最近在做算法作业时出现了错误,原因是没有弄清楚java集合的深度复制和浅度复 ...

  4. [No0000B9]C# 类型基础 值类型和引用类型 及其 对象复制 浅度复制vs深度复制 深入研究2

    接上[No0000B5]C# 类型基础 值类型和引用类型 及其 对象判等 深入研究1 对象复制 有的时候,创建一个对象可能会非常耗时,比如对象需要从远程数据库中获取数据来填充,又或者创建对象需要读取硬 ...

  5. 深度解析javascript中的浅复制和深复制

    原文:深度解析javascript中的浅复制和深复制 在谈javascript的浅复制和深复制之前,我们有必要在来讨论下js的数据类型.我们都知道有Number,Boolean,String,Null ...

  6. Java的深度克隆和浅度克隆

    说到克隆,其实是个比较简单的概念,跟现实生活正的克隆一样,复制一个一模一样的对象出来.clone()这个方法是从Object继承下来的,一个对象要实现克隆,需要实现一个叫做Cloneable的接口,这 ...

  7. List的深度copy和浅度拷贝

    List<Student> list= Arrays.asList( new Student("Fndroid", 22, Student.Sax.MALE, 180) ...

  8. Cloneable接口的作用与深度克隆与浅度克隆

    cloneable接口的作用 cloneable其实就是一个标记接口,只有实现这个接口后,然后在类中重写Object中的clone方法,然后通过类调用clone方法才能克隆成功,如果不实现这个接口,则 ...

  9. js中的深复制和浅复制

    在实际情况中经常会遇到对对象复制的问题.比如在处理项目中的一笔多结构的数据存储或者调用,这个时候你就要对对象(json)进行操作,而不同的操作根据不同的需求来定义.其中最常见最普遍的是对对象的复制,重 ...

随机推荐

  1. UVA-257 哈希算法

    UVA-257 题意: 给你很多串,你需要找到这个串内有没有两个长度大于3的回文字符串,且要保证这两个回文字符串不相同,也不能完全覆盖,但可以重合一部分 题解: 首先判断回文的话可以通过马拉车算法(M ...

  2. How many integers can you find HDU - 1796 容斥原理

    题意: 给你一个数n,找出来区间[1,n]内有多少书和n不互质 题解: 容斥原理 这一道题就让我真正了解容斥原理的实体部分 "容斥原理+枚举状态,碰到奇数加上(n-1)/lcm(a,b,c. ...

  3. CodeForces - 1100E 二分+拓扑排序

    题意: 一个n个节点的有向图,节点标号从1到n,存在m条单向边.每条单向边有一个权值,代表翻转其方向所需的代价.求使图变成无环图,其中翻转的最大边权值最小的方案,以及该方案翻转的最大的边权. Inpu ...

  4. Codeforces Round #643 (Div. 2) E. Restorer Distance (贪心,三分)

    题意:给你\(n\)个数,每次可以使某个数++,--,或使某个数--另一个++,分别消耗\(a,r,m\).求使所有数相同最少的消耗. 题解:因为答案不是单调的,所以不能二分,但不难发现,答案只有一个 ...

  5. Apple iOS 触控按钮 自动关闭 bug

    Apple iOS 触控按钮 自动关闭 bug bug 轻点 iPhone 背面可执行操作 您可以轻点两下或轻点三下 iPhone 背面以执行某些操作,如向上或向下滚动.截屏.打开"控制中心 ...

  6. Sentry React SourceMaps All In One

    Sentry React SourceMaps All In One React https://docs.sentry.io/platforms/javascript/guides/react/ h ...

  7. shit leetcode edge testcases

    shit leetcode edge testcases Merge Intervals try "use strict"; /** * * @author xgqfrms * @ ...

  8. React Hooks: useState All In One

    React Hooks: useState All In One useState import React, { useState } from 'react'; function Example( ...

  9. VSCode 开放式架构的产品实现思路

    VSCode 开放式架构的产品实现思路 https://code.visualstudio.com/ 源码 https://github.com/microsoft/vscode https://gi ...

  10. CSS 设置多行文本省略号 ...

    CSS 设置多行文本省略号 ... .box{ display: -webkit-box; overflow: hidden; text-overflow: ellipsis; word-wrap: ...