首页 > 编程语言 >深度复制:C# 中 List 与 List 多层嵌套不改变原值的实现方法

深度复制:C# 中 List 与 List 多层嵌套不改变原值的实现方法

时间:2024-03-04 09:04:17浏览次数:39  
标签:Info Name C# List 嵌套 复制 new 序列化

 

概述:以上内容详细介绍了在 C# 中实现不改变原 List 值的多层嵌套复制方法,包括使用 AutoMapper、Json.NET、以及对象序列化的步骤和示例。这些方法提供了灵活而高效的方式,可以根据项目需求选择最适合的深度复制方式。

1. 使用 AutoMapper 进行多层嵌套复制

AutoMapper 是一个对象映射工具,可以方便地进行对象之间的映射。以下是使用 AutoMapper 实现多层嵌套复制的步骤和示例:

首先,你需要在项目中安装 AutoMapper 包。你可以通过 NuGet 包管理器控制台运行以下命令来安装:

Install-Package AutoMapper

然后,你可以使用以下代码进行深度复制:

using AutoMapper;
using System;
using System.Collections.Generic;

class Person
{
    public string Name { get; set; }
    public int Age { get; set; }
}

class Student
{
    public string StudentId { get; set; }
    public Person Info { get; set; }
}

class Program
{
    static void Main()
    {
        // 创建原始 List,多层嵌套
        List<Student> originalList = new List<Student>
        {
            new Student { StudentId = "001", Info = new Person { Name = "Alice", Age = 25 } },
            new Student { StudentId = "002", Info = new Person { Name = "Bob", Age = 30 } }
        };

        // 使用 AutoMapper 实现深度复制
        List<Student> copiedList = DeepCopyWithAutoMapper(originalList);

        // 修改复制后的值
        copiedList[0].Info.Name = "Charlie";

        // 打印原始值,验证原始 List 的值是否改变
        Console.WriteLine("原始 List 的值:");
        PrintList(originalList);

        // 打印复制后的值
        Console.WriteLine("\n复制后 List 的值:");
        PrintList(copiedList);
    }

    static List<Student> DeepCopyWithAutoMapper(List<Student> originalList)
    {
        // 初始化 AutoMapper 配置
        var config = new MapperConfiguration(cfg =>
        {
            // 针对每一层嵌套的类型进行映射配置
            cfg.CreateMap<Student, Student>();
            cfg.CreateMap<Person, Person>();
        });

        // 创建映射器
        IMapper mapper = config.CreateMapper();

        // 使用映射器进行深度复制
        List<Student> newList = mapper.Map<List<Student>>(originalList);

        return newList;
    }

    // 打印 List 的方法
    static void PrintList(List<Student> list)
    {
        foreach (var student in list)
        {
            Console.WriteLine($"StudentId: {student.StudentId}, Name: {student.Info.Name}, Age: {student.Info.Age}");
        }
    }
}

在这个示例中,首先初始化 AutoMapper 配置,然后创建映射器,并使用映射器进行深度复制。

2. 使用 Json.NET 进行多层嵌套复制

Json.NET(Newtonsoft.Json)是一个用于处理 JSON 数据的强大库,也可以用于实现深度复制。以下是使用 Json.NET 实现多层嵌套复制的步骤和示例:

首先,你需要在项目中安装 Json.NET 包。你可以通过 NuGet 包管理器控制台运行以下命令来安装:

Install-Package Newtonsoft.Json

然后,你可以使用以下代码进行深度复制:

using Newtonsoft.Json;
using System;
using System.Collections.Generic;

class Person
{
    public string Name { get; set; }
    public int Age { get; set; }
}

class Student
{
    public string StudentId { get; set; }
    public Person Info { get; set; }
}

class Program
{
    static void Main()
    {
        // 创建原始 List,多层嵌套
        List<Student> originalList = new List<Student>
        {
            new Student { StudentId = "001", Info = new Person { Name = "Alice", Age = 25 } },
            new Student { StudentId = "002", Info = new Person { Name = "Bob", Age = 30 } }
        };

        // 使用 Json.NET 实现深度复制
        List<Student> copiedList = DeepCopyWithJson(originalList);

        // 修改复制后的值
        copiedList[0].Info.Name = "Charlie";

        // 打印原始值,验证原始 List 的值是否改变
        Console.WriteLine("原始 List 的值:");
        PrintList(originalList);

        // 打印复制后的值
        Console.WriteLine("\n复制后 List 的值:");
        PrintList(copiedList);
    }

    static List<Student> DeepCopyWithJson(List<Student> originalList)
    {
        // 使用 JsonConvert 进行深度复制
        string json = JsonConvert.SerializeObject(originalList);
        List<Student> newList = JsonConvert.DeserializeObject<List<Student>>(json);

        return newList;
    }

    // 打印 List 的方法
    static void PrintList(List<Student> list)
    {
        foreach

 (var student in list)
        {
            Console.WriteLine($"StudentId: {student.StudentId}, Name: {student.Info.Name}, Age: {student.Info.Age}");
        }
    }
}

在这个示例中,使用 JsonConvert 将原始 List 转换为 JSON 字符串,然后再从 JSON 字符串中反序列化得到新的 List,实现了深度复制。

3. 使用对象序列化和反序列化进行深度复制

另一种常见的方法是使用 C# 的对象序列化和反序列化功能,将对象序列化为字节流,然后再反序列化为新的对象。以下是使用序列化和反序列化实现多层嵌套复制的步骤和示例:

using System;
using System.Collections.Generic;
using System.IO;
using System.Runtime.Serialization;
using System.Runtime.Serialization.Formatters.Binary;

class Person
{
    public string Name { get; set; }
    public int Age { get; set; }
}

class Student
{
    public string StudentId { get; set; }
    public Person Info { get; set; }
}

class Program
{
    static void Main()
    {
        // 创建原始 List,多层嵌套
        List<Student> originalList = new List<Student>
        {
            new Student { StudentId = "001", Info = new Person { Name = "Alice", Age = 25 } },
            new Student { StudentId = "002", Info = new Person { Name = "Bob", Age = 30 } }
        };

        // 使用序列化和反序列化实现深度复制
        List<Student> copiedList = DeepCopyWithSerialization(originalList);

        // 修改复制后的值
        copiedList[0].Info.Name = "Charlie";

        // 打印原始值,验证原始 List 的值是否改变
        Console.WriteLine("原始 List 的值:");
        PrintList(originalList);

        // 打印复制后的值
        Console.WriteLine("\n复制后 List 的值:");
        PrintList(copiedList);
    }

    static List<Student> DeepCopyWithSerialization(List<Student> originalList)
    {
        IFormatter formatter = new BinaryFormatter();
        using (MemoryStream stream = new MemoryStream())
        {
            formatter.Serialize(stream, originalList);
            stream.Seek(0, SeekOrigin.Begin);
            return (List<Student>)formatter.Deserialize(stream);
        }
    }

    // 打印 List 的方法
    static void PrintList(List<Student> list)
    {
        foreach (var student in list)
        {
            Console.WriteLine($"StudentId: {student.StudentId}, Name: {student.Info.Name}, Age: {student.Info.Age}");
        }
    }
}

在这个示例中,使用 BinaryFormatter 将原始 List 序列化为字节流,然后再反序列化得到新的 List,实现了深度复制。

 

标签:Info,Name,C#,List,嵌套,复制,new,序列化
From: https://www.cnblogs.com/hanbing81868164/p/18051098

相关文章

  • 精简实现、线程安全:C#通用单例泛型基类助你轻松创建单例模式
     概述:该通用单例泛型基类使用C#实现,线程安全,通过泛型参数和Lazy<T>实现简化的单例模式。优点包括线程安全、泛型通用性、简化实现、以及延迟加载的特性。优点:线程安全: 使用Lazy<T>确保了线程安全的延迟初始化,避免了在多线程环境下可能导致的竞态条件问题。泛型通用性: 通......
  • C#/.NET/.NET Core优秀项目和框架2024年2月简报
    前言公众号每月定期推广和分享的C#/.NET/.NETCore优秀项目和框架(每周至少会推荐两个优秀的项目和框架当然节假日除外),公众号推文中有项目和框架的介绍、功能特点、使用方式以及部分功能截图等(打不开或者打开GitHub很慢的同学可以优先查看公众号推文,文末一定会附带项目和框架源码......
  • ascii码表里 转义字符有哪些
    ASCII码表中的转义字符主要有以下几种:\a:响铃(BEL),对应的ASCII码值是7。\b:退格(BS),对应的ASCII码值是8。它将光标移回前一个字符,但不会删除该字符,只是将光标移回。\f:换页(FF),对应的ASCII码值是12。它用于控制输出设备(如打印机)开始新的页面。\n:换行(LF),对应的ASCII码值是10。它......
  • ElasticSearch入门实战
    全文检索根据搜索条件将相关内容结果进行检索。查询:根据明确的条件对数据进行过滤,例如年龄>20且工资<10000的人进行过略。检索:没有明确的条件,通过相关性将结果进行返回,例如根据查询条件的同义词、错别字等等进行查询。通过常规的模糊查询只能查出包含xxx的内容,而没办......
  • northwest uniapp: 顶部导航栏右侧添加uni-icons图标,并绑定点击事
    说明该配置按页面配置,无法全局配置使用方法引入uni-icons插件插件地址:点击此处跳转到uni-icons引入以后,会发现uni-modules中新增如下目录和文件:仅需要那个uniicons.ttf文件,不引入插件、单独把那个文件下载到本地也是可以的。复制uniicons.ttf文件到static目录下不知道......
  • 【C++ STL容器set 】std::set 的全方位解析
    装载自知乎(虽然有AI辅助操作,但是确实写得好好):【C++STL容器set】std::set的全方位解析-知乎(zhihu.com)<imgsrc="https://pic3.zhimg.com/v2-cc8068b8931c7f65e9a89717e2ab404e_b.jpg"data-size="normal"data-rawwidth="1024"data-rawheight="1024......
  • RUN、CMD、ENTRYPOINT区别
    1、简要概述RUN:执行命令并创建新的镜像层,RUN经常用于安装软件包。CMD:设置容器启动后默认执行的命令及其参数,但CD能够被dockerrun后面跟的命令行参数替换。ENTRYPOINT:配置容器启动时运行的命令。2.Shell和Exec格式两种方式指定RUN、CMD和ENTRYPOINT要运行的命......
  • CAS解析
    目录什么是CAS应用场景以AtomicInteger为例,分析在Java中如何利用CAS实现原子操作的Unsafe类Unsafe类中的compareAndSwapInt缺点什么是CAS应用场景:并发我认为V的值应该是A,如果是的话,那我就把它修改成B,如果不是A(说明被别人修改过了),那我就不修改了,避免多人同时操作导致出错......
  • 4_C# 中的 ref 关键字有什么作用
    C#中的ref关键字有什么作用?参数在使用ref关键字进行引用传递时,必须在方法调用之前对其进行初始化。ref关键字既可以在进入方法之前初始化参数的值,也可以在方法内部对参数进行修改。ref参数在进入方法时保持原始值,并在方法结束后将值带回到调用处。ref关键字的作用:......
  • 1_readonly与const区别
    readonly与const区别?readonly关键字(运行时常量):字段可以在声明或构造函数中初始化,常作为运行时常量使用。const关键字(编译时常量):字段只能在该字段的声明时初始化,常作为编译时常量使用过。C#中的readonly和const都是用于声明常量的关键字,但它们之间存在一些关键的区......