热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

C#深拷贝、浅拷贝

说明浅拷贝指的是拷贝一个对象时,仅仅拷贝对象的引用,即两个对象还是引用同一份实体。此时,其中一个对象的改变会影响到另一个对象。深拷贝

说明


浅拷贝

指的是拷贝一个对象时,仅仅拷贝对象的引用,即两个对象还是引用同一份实体。此时,其中一个对象的改变会影响到另一个对象。

深拷贝

指的是拷贝一个对象时,不仅仅把对象的引用进行复制,还把该对象引用的值也一起拷贝。这样拷贝后的对象就和源对象互相独立,其中任何一个对象的改动都不会对另外一个对象造成影响。

实现方式


浅拷贝

对象继承接口ICloneable,重写Clone方法。
伪代码如下:

public class classA : ICloneable{public object Clone(){return this.MemberwiseClone();}}

上面代码中使用.NET提供的MemberwiseClone()方法来实现浅拷贝。

深拷贝

实现深拷贝,有以下方式:
1,上述Clone()方法体中,不用浅拷贝的MemberwiseClone()方法,通过重新开辟对象内存,以及重新开辟对象引用对象成员内存的方式,来实现深拷贝。
2,序列化,反序列化的方式

// 利用XML序列化和反序列化实现public static T DeepCopyWithXmlSerializer(T obj){object retval;using (MemoryStream ms = new MemoryStream()){XmlSerializer xml = new XmlSerializer(typeof(T));xml.Serialize(ms, obj);ms.Seek(0, SeekOrigin.Begin);retval = xml.Deserialize(ms);ms.Close();}return (T)retval;}// 利用二进制序列化和反序列实现public static T DeepCopyWithBinarySerialize(T obj){object retval;using (MemoryStream ms = new MemoryStream()){BinaryFormatter bf = new BinaryFormatter();// 序列化成流bf.Serialize(ms, obj);ms.Seek(0, SeekOrigin.Begin);// 反序列化成对象retval = bf.Deserialize(ms);ms.Close();}return (T)retval;}

3,反射的方式
下述两种反射方式,要求:实体类有个无参的构造方法。且都没有解决类和类之间循环引用的问题

///

/// 对象拷贝/// /// 被复制对象/// 新对象private object CopyOjbect(object obj) {if (obj == null) {return null;}Object targetDeepCopyObj;Type targetType = obj.GetType();//值类型 if (targetType.IsValueType == true|| targetType.FullName == "System.String") {targetDeepCopyObj = obj;}//引用类型 else {targetDeepCopyObj = System.Activator.CreateInstance(targetType); //创建引用对象 System.Reflection.MemberInfo[] memberCollection = obj.GetType().GetMembers();foreach (System.Reflection.MemberInfo member in memberCollection) {//拷贝字段if (member.MemberType == System.Reflection.MemberTypes.Field){System.Reflection.FieldInfo field = (System.Reflection.FieldInfo)member;Object fieldValue = field.GetValue(obj);if (fieldValue is ICloneable){field.SetValue(targetDeepCopyObj, (fieldValue as ICloneable).Clone());}else{field.SetValue(targetDeepCopyObj, CopyOjbect(fieldValue));}}//拷贝属性else if (member.MemberType == System.Reflection.MemberTypes.Property) {System.Reflection.PropertyInfo myProperty = (System.Reflection.PropertyInfo)member;MethodInfo info = myProperty.GetSetMethod(false);if (info != null) {try {object propertyValue = myProperty.GetValue(obj, null);if (propertyValue is ICloneable) {myProperty.SetValue(targetDeepCopyObj, (propertyValue as ICloneable).Clone(), null);}else {myProperty.SetValue(targetDeepCopyObj, CopyOjbect(propertyValue), null);}}catch (System.Exception ex) {}}}}}return targetDeepCopyObj;}

// 利用反射实现深拷贝public static T DeepCopyWithReflection(T obj){Type type = obj.GetType();// 如果是字符串或值类型则直接返回if (obj is string || type.IsValueType) return obj;if (type.IsArray){Type elementType = Type.GetType(type.FullName.Replace("[]", string.Empty));var array = obj as Array;Array copied = Array.CreateInstance(elementType, array.Length);for (int i = 0; i

补充

下面代码,通过引用赋值和new重新分配内存的方式,完成了s–>s1和s–>s2的深拷贝。
Student s = new Student(“张成”, 21);
Student s1 = s;
s = new Student(“郭立强”, 32);
Student s2 = s;

在这里插入图片描述


推荐阅读
  • 向QTextEdit拖放文件的方法及实现步骤
    本文介绍了在使用QTextEdit时如何实现拖放文件的功能,包括相关的方法和实现步骤。通过重写dragEnterEvent和dropEvent函数,并结合QMimeData和QUrl等类,可以轻松实现向QTextEdit拖放文件的功能。详细的代码实现和说明可以参考本文提供的示例代码。 ... [详细]
  • 本文分享了一个关于在C#中使用异步代码的问题,作者在控制台中运行时代码正常工作,但在Windows窗体中却无法正常工作。作者尝试搜索局域网上的主机,但在窗体中计数器没有减少。文章提供了相关的代码和解决思路。 ... [详细]
  • Java序列化对象传给PHP的方法及原理解析
    本文介绍了Java序列化对象传给PHP的方法及原理,包括Java对象传递的方式、序列化的方式、PHP中的序列化用法介绍、Java是否能反序列化PHP的数据、Java序列化的原理以及解决Java序列化中的问题。同时还解释了序列化的概念和作用,以及代码执行序列化所需要的权限。最后指出,序列化会将对象实例的所有字段都进行序列化,使得数据能够被表示为实例的序列化数据,但只有能够解释该格式的代码才能够确定数据的内容。 ... [详细]
  • [大整数乘法] java代码实现
    本文介绍了使用java代码实现大整数乘法的过程,同时也涉及到大整数加法和大整数减法的计算方法。通过分治算法来提高计算效率,并对算法的时间复杂度进行了研究。详细代码实现请参考文章链接。 ... [详细]
  • 本文介绍了iOS数据库Sqlite的SQL语句分类和常见约束关键字。SQL语句分为DDL、DML和DQL三种类型,其中DDL语句用于定义、删除和修改数据表,关键字包括create、drop和alter。常见约束关键字包括if not exists、if exists、primary key、autoincrement、not null和default。此外,还介绍了常见的数据库数据类型,包括integer、text和real。 ... [详细]
  • 本文介绍了在iOS开发中使用UITextField实现字符限制的方法,包括利用代理方法和使用BNTextField-Limit库的实现策略。通过这些方法,开发者可以方便地限制UITextField的字符个数和输入规则。 ... [详细]
  • 纠正网上的错误:自定义一个类叫java.lang.System/String的方法
    本文纠正了网上关于自定义一个类叫java.lang.System/String的错误答案,并详细解释了为什么这种方法是错误的。作者指出,虽然双亲委托机制确实可以阻止自定义的System类被加载,但通过自定义一个特殊的类加载器,可以绕过双亲委托机制,达到自定义System类的目的。作者呼吁读者对网上的内容持怀疑态度,并带着问题来阅读文章。 ... [详细]
  • 本文介绍了在MFC下利用C++和MFC的特性动态创建窗口的方法,包括继承现有的MFC类并加以改造、插入工具栏和状态栏对象的声明等。同时还提到了窗口销毁的处理方法。本文详细介绍了实现方法并给出了相关注意事项。 ... [详细]
  • SpringBoot uri统一权限管理的实现方法及步骤详解
    本文详细介绍了SpringBoot中实现uri统一权限管理的方法,包括表结构定义、自动统计URI并自动删除脏数据、程序启动加载等步骤。通过该方法可以提高系统的安全性,实现对系统任意接口的权限拦截验证。 ... [详细]
  • 本文介绍了Java高并发程序设计中线程安全的概念与synchronized关键字的使用。通过一个计数器的例子,演示了多线程同时对变量进行累加操作时可能出现的问题。最终值会小于预期的原因是因为两个线程同时对变量进行写入时,其中一个线程的结果会覆盖另一个线程的结果。为了解决这个问题,可以使用synchronized关键字来保证线程安全。 ... [详细]
  • 个人学习使用:谨慎参考1Client类importcom.thoughtworks.gauge.Step;importcom.thoughtworks.gauge.T ... [详细]
  • 本文讨论了Kotlin中扩展函数的一些惯用用法以及其合理性。作者认为在某些情况下,定义扩展函数没有意义,但官方的编码约定支持这种方式。文章还介绍了在类之外定义扩展函数的具体用法,并讨论了避免使用扩展函数的边缘情况。作者提出了对于扩展函数的合理性的质疑,并给出了自己的反驳。最后,文章强调了在编写Kotlin代码时可以自由地使用扩展函数的重要性。 ... [详细]
  • 怎么在PHP项目中实现一个HTTP断点续传功能发布时间:2021-01-1916:26:06来源:亿速云阅读:96作者:Le ... [详细]
  • 本文介绍了RxJava在Android开发中的广泛应用以及其在事件总线(Event Bus)实现中的使用方法。RxJava是一种基于观察者模式的异步java库,可以提高开发效率、降低维护成本。通过RxJava,开发者可以实现事件的异步处理和链式操作。对于已经具备RxJava基础的开发者来说,本文将详细介绍如何利用RxJava实现事件总线,并提供了使用建议。 ... [详细]
  • 本文介绍了在Cpp中将字符串形式的数值转换为int或float等数值类型的方法,主要使用了strtol、strtod和strtoul函数。这些函数可以将以null结尾的字符串转换为long int、double或unsigned long类型的数值,且支持任意进制的字符串转换。相比之下,atoi函数只能转换十进制数值且没有错误返回。 ... [详细]
author-avatar
Z-19932502872833
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有