克隆对象在开发过程中经常会遇到,有些时候需要浅克隆,有些时候需要深克隆,具体它们之间有什么区别,以及实现方式有哪些,在这里总结一下。
实现深克隆有以下几种方法。
手动
代码如下:
1 2 3 4 5 6 7 8 9 10 | //手动复制 var user2 = new User { Id = user1.Id, Name = new UserName { FirstName= user1.Name.FirstName, LastName= user1.Name.LastName } }; |
反射
代码如下:
1 //反射 2 var user3 = user1.Copy() as User;
扩展方法:

1 public static class DeepCopyHelper 2 { 3 public static object Copy(this object obj) 4 { 5 Object targetDeepCopyObj; 6 Type targetType = obj.GetType(); 7 //值类型 8 if (targetType.IsValueType == true) 9 { 10 targetDeepCopyObj = obj; 11 } 12 //引用类型 13 else 14 { 15 targetDeepCopyObj = System.Activator.CreateInstance(targetType); //创建引用对象 16 System.Reflection.MemberInfo[] memberCollection = obj.GetType().GetMembers(); 17 18 foreach (System.Reflection.MemberInfo member in memberCollection) 19 { 20 if (member.MemberType == System.Reflection.MemberTypes.Field) 21 { 22 System.Reflection.FieldInfo field = (System.Reflection.FieldInfo)member; 23 Object fieldValue = field.GetValue(obj); 24 if (fieldValue is ICloneable) 25 { 26 field.SetValue(targetDeepCopyObj, (fieldValue as ICloneable).Clone()); 27 } 28 else 29 { 30 field.SetValue(targetDeepCopyObj, Copy(fieldValue)); 31 } 32 33 } 34 else if (member.MemberType == System.Reflection.MemberTypes.Property) 35 { 36 System.Reflection.PropertyInfo myProperty = (System.Reflection.PropertyInfo)member; 37 MethodInfo info = myProperty.GetSetMethod(false); 38 if (info != null) 39 { 40 object propertyValue = myProperty.GetValue(obj, null); 41 if (propertyValue is ICloneable) 42 { 43 myProperty.SetValue(targetDeepCopyObj, (propertyValue as ICloneable).Clone(), null); 44 } 45 else 46 { 47 myProperty.SetValue(targetDeepCopyObj, Copy(propertyValue), null); 48 } 49 } 50 51 } 52 } 53 } 54 return targetDeepCopyObj; 55 } 56 }
序列化
代码如下:
1 //序列化 2 var user4 = user1.DeepClone();
扩展方法:

1 /// <summary> 2 /// 深克隆 3 /// 先序列化再反序列化 4 /// </summary> 5 /// <typeparam name="T"></typeparam> 6 /// <param name="obj"></param> 7 /// <returns></returns> 8 public static T DeepClone<T>(this T obj) where T : class 9 { 10 return obj != null ? obj.ToJson().FromJson<T>() : null; 11 }
其它还有使用表达式。
总结:
- 手动复制性能最好,但是遇到很复杂的类的时候,工作量很大。
- 反射和序列化比起来,序列化更简单。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 10年+ .NET Coder 心语 ── 封装的思维:从隐藏、稳定开始理解其本质意义
· 地球OL攻略 —— 某应届生求职总结
· 提示词工程——AI应用必不可少的技术
· Open-Sora 2.0 重磅开源!
· 周边上新:园子的第一款马克杯温暖上架
2013-06-05 Ext Js MVC系列二 利用Application和Viewport进行应用程序初始化和页面布局