您的位置:首页 >.NET中的深拷贝实现方法详解
发布于2026-05-03 阅读(0)
扫一扫,手机访问
咱们先明确一下概念:深拷贝要做的,可不是简单地复制一个引用指针。它意味着创建一个全新的对象,并且像“开枝散叶”一样,递归地复制原对象及其内部所有引用对象的值。这和浅拷贝只复制“表面”有着本质区别。
这招算是“经典流派”了,通过把对象序列化成二进制流,再反序列化回来,自然就得到了一个全新的副本。但有个前提,你的相关类型都得是可序列化的。
using System;
using System.IO;
using System.Runtime.Serialization;
using System.Runtime.Serialization.Formatters.Binary;
public static class ObjectCopier
{
public static T DeepCopy(T obj)
{
if (!typeof(T).IsSerializable)
{
throw new ArgumentException("The type must be serializable.", nameof(obj));
}
if (ReferenceEquals(obj, null))
{
return default;
}
IFormatter formatter = new BinaryFormatter();
using (var stream = new MemoryStream())
{
formatter.Serialize(stream, obj);
stream.Seek(0, SeekOrigin.Begin);
return (T)formatter.Deserialize(stream);
}
}
}
随着JSON的普及,这方法现在用得越来越多了。它本质上是把对象“翻译”成JSON文本,然后再“翻译”回来,实现拷贝。两种主流库都可以用,选择哪个看你的项目环境。
// 使用 Newtonsoft.Json using Newtonsoft.Json; public static T DeepCopy(T obj) { var json = JsonConvert.SerializeObject(obj); return JsonConvert.DeserializeObject (json); } // 使用 System.Text.Json (推荐.NET Core 3.0+) using System.Text.Json; public static T DeepCopy (T obj) { var json = JsonSerializer.Serialize(obj); return JsonSerializer.Deserialize (json); }
如果你需要更精细的控制,或者对象结构特殊,手动实现ICloneable接口是个可靠的选择。这种方式需要你亲自动手,为每个引用类型成员安排“克隆”工作,灵活性最高。
public class MyClass : ICloneable
{
public int Value { get; set; }
public MyOtherClass Other { get; set; }
public object Clone()
{
var copy = (MyClass)MemberwiseClone(); // 先来个浅拷贝
copy.Other = (MyOtherClass)Other.Clone(); // 再手动深拷贝引用成员
return copy;
}
}
public class MyOtherClass : ICloneable
{
public string Name { get; set; }
public object Clone()
{
return MemberwiseClone(); // 如果成员都是值类型,浅拷贝就够了
}
}
当面对属性众多、层次复杂的对象图时,配置一次AutoMapper,之后就可以优雅地一键深拷贝了。它特别适合在项目已经引入该库,或者对象结构非常规整的场景。
using AutoMapper;
var config = new MapperConfiguration(cfg =>
{
cfg.CreateMap();
cfg.CreateMap();
});
var mapper = config.CreateMapper();
var copy = mapper.Map(original);
方法虽多,但各有利弊,选择时别忘了下面这些潜在的“坑”:
[Serializable]特性,要么能被JSON序列化器处理)。说了这么多,到底该怎么选?这里有个简单的决策思路:
System.Text.Json,在.NET Core环境下性能表现更佳)。AutoMapper或者老老实实实现ICloneable接口,以获得更好的可控性。总而言之,没有绝对最好的方法,只有最适合当前场景的选择。综合考量你的具体需求、对象结构的复杂程度以及对性能的要求,才能做出最合适的决定。
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
售后无忧
立即购买>office旗舰店
正版软件
正版软件
正版软件
正版软件
正版软件
1
2
3
7
9