.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
  • 对于性能敏感场景:考虑手动实现深拷贝逻辑

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

相关推荐
SuperEugene1 分钟前
Axios 接口请求规范实战:请求参数 / 响应处理 / 异常兜底,避坑中后台 API 调用混乱|API 与异步请求规范篇
开发语言·前端·javascript·vue.js·前端框架·axios
xuxie991 小时前
N11 ARM-irq
java·开发语言
cjy0001111 小时前
springboot的 nacos 配置获取不到导致启动失败及日志不输出问题
java·spring boot·后端
唐青枫1 小时前
C#.NET ReaderWriterLockSlim 深入解析:读写锁原理、升级锁与使用边界
c#·.net
wefly20172 小时前
从使用到原理,深度解析m3u8live.cn—— 基于 HLS.js 的 M3U8 在线播放器实现
java·开发语言·前端·javascript·ecmascript·php·m3u8
zhenxin01222 小时前
Spring Boot实现定时任务
java
小江的记录本2 小时前
【事务】Spring Framework核心——事务管理:ACID特性、隔离级别、传播行为、@Transactional底层原理、失效场景
java·数据库·分布式·后端·sql·spring·面试
sheji34162 小时前
【开题答辩全过程】以 基于springboot的校园失物招领系统为例,包含答辩的问题和答案
java·spring boot·后端
luanma1509802 小时前
PHP vs C++:编程语言终极对决
开发语言·c++·php
寂静or沉默2 小时前
2026最新Java岗位从P5-P7的成长面试进阶资源分享!
java·开发语言·面试