.NET 中的深拷贝实现方法

在 .NET 中实现深拷贝(Deep Copy)有几种常用方法,深拷贝是指创建一个新对象,并递归地复制原对象及其所有引用对象,而不仅仅是复制引用。

目录

  • [1. 使用序列化/反序列化](#1. 使用序列化/反序列化)
  • [2. 使用 JSON 序列化(Newtonsoft.Json 或 System.Text.Json)](#2. 使用 JSON 序列化(Newtonsoft.Json 或 System.Text.Json))
  • [3. 实现 ICloneable 接口(手动实现)](#3. 实现 ICloneable 接口(手动实现))
  • [4. 使用 AutoMapper(适用于复杂对象)](#4. 使用 AutoMapper(适用于复杂对象))
  • [5. 注意事项](#5. 注意事项)
  • [6. 推荐方法](#6. 推荐方法)

1. 使用序列化/反序列化

csharp 复制代码
using System;
using System.IO;
using System.Runtime.Serialization;
using System.Runtime.Serialization.Formatters.Binary;

public static class ObjectCopier
{
    public static T DeepCopy<T>(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);
        }
    }
}

2. 使用 JSON 序列化(Newtonsoft.Json 或 System.Text.Json)

csharp 复制代码
// 使用 Newtonsoft.Json
using Newtonsoft.Json;

public static T DeepCopy<T>(T obj)
{
    var json = JsonConvert.SerializeObject(obj);
    return JsonConvert.DeserializeObject<T>(json);
}

// 使用 System.Text.Json (推荐.NET Core 3.0+)
using System.Text.Json;

public static T DeepCopy<T>(T obj)
{
    var json = JsonSerializer.Serialize(obj);
    return JsonSerializer.Deserialize<T>(json);
}

3. 实现 ICloneable 接口(手动实现)

csharp 复制代码
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(); // 浅拷贝(因为只有值类型)
    }
}

4. 使用 AutoMapper(适用于复杂对象)

csharp 复制代码
using AutoMapper;

var config = new MapperConfiguration(cfg =>
{
    cfg.CreateMap<MyClass, MyClass>();
    cfg.CreateMap<MyOtherClass, MyOtherClass>();
});

var mapper = config.CreateMapper();
var copy = mapper.Map<MyClass>(original);

5. 注意事项

  1. 序列化方法要求所有相关类都是可序列化的(有 [Serializable] 特性或可以被 JSON 序列化)
  2. 循环引用可能导致堆栈溢出或序列化异常
  3. 性能考虑:对于大型对象图,序列化方法可能较慢
  4. 某些特殊类型(如委托、COM 对象)可能无法正确拷贝

6. 推荐方法

  • 对于简单对象:使用 JSON 序列化(System.Text.Json 性能较好)
  • 对于复杂对象图:考虑实现 ICloneable 或使用 AutoMapper
  • 对于性能敏感场景:考虑手动实现深拷贝逻辑

选择哪种方法取决于具体需求、对象复杂度和性能要求。

相关推荐
皮皮林5518 小时前
IDEA 源码阅读利器,你居然还不会?
java·intellij idea
该用户已不存在12 小时前
6个值得收藏的.NET ORM 框架
前端·后端·.net
卡尔特斯12 小时前
Android Kotlin 项目代理配置【详细步骤(可选)】
android·java·kotlin
白鲸开源12 小时前
Ubuntu 22 下 DolphinScheduler 3.x 伪集群部署实录
java·ubuntu·开源
ytadpole12 小时前
Java 25 新特性 更简洁、更高效、更现代
java·后端
纪莫12 小时前
A公司一面:类加载的过程是怎么样的? 双亲委派的优点和缺点? 产生fullGC的情况有哪些? spring的动态代理有哪些?区别是什么? 如何排查CPU使用率过高?
java·java面试⑧股
JavaGuide13 小时前
JDK 25(长期支持版) 发布,新特性解读!
java·后端
用户37215742613513 小时前
Java 轻松批量替换 Word 文档文字内容
java
白鲸开源13 小时前
教你数分钟内创建并运行一个 DolphinScheduler Workflow!
java
Java中文社群14 小时前
有点意思!Java8后最有用新特性排行榜!
java·后端·面试