在 .NET 中实现对象的深度克隆,不能简单使用 MemberwiseClone(),因为那只是浅拷贝。深度克隆意味着新对象和原对象完全独立,包括它们内部引用的所有对象也都被复制,而不是共享引用。以下是几种常用的深度克隆方法:
1. 使用序列化(推荐通用方式)
通过序列化和反序列化实现深度克隆,是最常见且可靠的方法之一。只要对象是可序列化的,就能轻松完成深拷贝。二进制序列化(.NET Framework)
适用于 .NET Framework 项目(注意:.NET 5+ 不支持 BinaryFormatter):using System;
using System.IO;
using System.Runtime.Serialization.Formatters.Binary;
public static T DeepClone<T>(T obj)
{
if (obj == null) return default(T);
using (var ms = new MemoryStream())
{
var formatter = new BinaryFormatter();
formatter.Serialize(ms, obj);
ms.Position = 0;
return (T)formatter.Deserialize(ms);
}
}
JSON 序列化(跨平台推荐)
适用于 .NET Core / .NET 5+,更安全、跨平台:using System.Text.Json;
public static T DeepCloneJson<T>(T obj)
{
var json = JsonSerializer.Serialize(obj);
return JsonSerializer.Deserialize<T>(json);
}
要求:对象的属性需为 public,且有 getter/setter。复杂类型需支持序列化。
2. 实现 ICloneable 接口(手动控制)
适用于需要精细控制克隆逻辑的场景。public class Person : ICloneable
{
public string Name { get; set; }
public Address Address { get; set; }
public object Clone()
{
var clone = new Person
{
Name = this.Name,
Address = this.Address?.Clone() as Address // 假设 Address 也实现了 Clone
};
return clone;
}
}
public class Address : ICloneable
{
public string City { get; set; }
public object Clone() => new Address { City = this.City };
}
优点是性能好、可控性强;缺点是代码量大,嵌套多时维护麻烦。
3. 使用第三方库(高效便捷)
比如 AutoMapper 或 FastDeepCloner 等库可以简化操作。示例:使用 FastDeepCloner(NuGet 包)
Install-Package FastDeepCloner使用:
var clonedObj = DeepCloner.Clone(originalObj);性能优秀,语法简洁,适合复杂对象。
4. 表达式树或 IL 生成(高级方案)
某些高性能场景下,可通过表达式树或 Emit 动态生成拷贝代码,但实现复杂,一般由框架处理。基本上就这些常用方式。选择哪种取决于你的运行环境(.NET Framework 还是 .NET 5+)、性能要求和对象结构复杂度。对于大多数情况,JSON 序列化已经足够安全且易用。
