C#中的对象深拷贝和浅拷贝

· 浏览次数 : 0

小编点评

**C# 中对象深拷贝和浅拷贝概述** **浅拷贝** * 只复制对象的值类型成员。 * 引用类型成员的引用则保持不变。 * 效率更高,但可能导致无法避免意外修改原始对象的异常。 **深拷贝** * 不仅复制对象的值类型成员,而且还复制所有引用类型成员的实际对象。 * 效率相对低,但可以避免无法避免意外修改原始对象的异常。 **示例代码** **浅拷贝** ```csharp public static class Person { public string Name { get; set; } public int Age { get; set; } public Address Address { get; set; } public Person Clone() { return (Person)this.MemberwiseClone(); } } ``` **深拷贝** ```csharp public static class Person { public string Name { get; set; } public int Age { get; set; } public Address Address { get; set; } public static void Copy2() { Person person1 = new Person() { Name = "张三", Address = new Address() { City = "北京" } }; // 深拷贝1 反射 Person person2 = JsonConvert.DeserializeObject(JsonConvert.SerializeObject(person1)); // 修改原对象的属性 person1.Address.City = "上海"; person2.Name = "李四"; person2.Address.City = "昆明"; Console.WriteLine("原对象和副本不相互影响。"); } } ``` **结论** 选择使用浅拷贝还是深拷贝取决于需要的数据传输效率和避免意外修改原始对象的风险。

正文

C#中的对象深拷贝和浅拷贝

概述

在C#中,对象拷贝是指将一个对象的副本创建到另一个对象中。对象拷贝通常用于数据传输或创建对象的新实例。

C#中有两种主要的拷贝方式:浅拷贝和深拷贝

1. 浅拷贝

浅拷贝是指只拷贝对象的值类型成员,而引用类型成员的引用则保持不变。这意味着新的对象和原始对象将共享所有引用类型成员的实际对象。

实现方式

  • this.MemberwiseClone();

示例代码

实体

public class Person
{
    public Person()
    {
        this.Address = new Address();
    }
    public string Name { get; set; }
     public int Age { get; set; }
    public Address Address { get; set; }

    public Person Clone()
    {
        return (Person)this.MemberwiseClone();
    }
}

public class Address
{
    public string Street { get; set; }
    public string City { get; set; }
}

调用

 Person person1 = new Person()
 {
     Name = "张三",
     Address = new Address()
     {
         City = "北京",
     }
 };

 Person person2 = person1.Clone();//浅拷贝
 //修改原对象的属性
 person1.Address.City = "上海";
 //修改副本对象的属性
 person2.Name = "李四";
 person2.Address.City = "昆明";
 string result = $"原对象{JsonConvert.SerializeObject(person1)}。副本{JsonConvert.SerializeObject(person2)}";
 MessageBox.Show($"浅拷贝:原对象和副本修改引用类型属性后相互影响。{result}");

2. 深拷贝

深拷贝是指不仅拷贝对象的值类型成员,而且还拷贝所有引用类型成员的实际对象。这意味着新的对象将拥有其引用类型成员的完全独立副本。

实现方式

  • 反射
  • 序列化
  • 对象映射(三方开源如TinyMapper、AutoMapper)。

示例代码

/// <summary>
/// 深拷贝
/// </summary>
public static void Copy2()
{
    Person person1 = new Person()
    {
        Name = "张三",
        Address = new Address()
        {
            City = "北京",
        }
    };

    //Person person2 = CreateDeepCopy(person1);//深拷贝1反射
    // Person person2 =JsonConvert.DeserializeObject<Person>(JsonConvert.SerializeObject(person1));//深拷贝2序列化
    Person person2 =person1.MapTo<Person,Person>();//深拷贝3对象映射
    //修改原对象的属性
    person1.Address.City = "上海";
    //修改副本对象的属性
    person2.Name = "李四";
    person2.Address.City = "昆明";
    string result = $"原对象{JsonConvert.SerializeObject(person1)}。副本{JsonConvert.SerializeObject(person2)}";
    MessageBox.Show($"深拷贝:原对象和副本不相互影响。{result}");
}


/// <summary>
/// 使用反射进行深拷贝
/// </summary>
/// <typeparam name="T"></typeparam>
/// <param name="original"></param>
/// <returns></returns>
static T CreateDeepCopy<T>(T original)
{
    if (original == null)
    {
        return default(T);
    }

    Type type = original.GetType();
    object newObject = Activator.CreateInstance(type);

    foreach (FieldInfo fieldInfo in type.GetFields())
    {
        if (fieldInfo.IsStatic)
        {
            continue;
        }

        object value = fieldInfo.GetValue(original);
        fieldInfo.SetValue(newObject, CreateDeepCopy(value));
    }

    return (T)newObject;
}

总结

浅拷贝通常用于数据传输,因为它是快速且有效的。但是,如果需要避免意外修改原始对象,则应使用深拷贝。

以下是一些有关何时使用浅拷贝和深拷贝的准则:

  • 使用浅拷贝:
    • 当需要快速创建对象副本时
    • 当原始对象不可变时
    • 当原始对象和副本不会同时使用时
  • 使用深拷贝:
    • 当需要避免意外修改原始对象时
    • 当原始对象和副本需要同时使用时
    • 当原始对象包含引用类型成员时

引用

与C#中的对象深拷贝和浅拷贝相似的内容:

C#中的对象深拷贝和浅拷贝

目录C#中的对象深拷贝和浅拷贝概述1. 浅拷贝2. 深拷贝总结引用 C#中的对象深拷贝和浅拷贝 概述 在C#中,对象拷贝是指将一个对象的副本创建到另一个对象中。对象拷贝通常用于数据传输或创建对象的新实例。 C#中有两种主要的拷贝方式:浅拷贝和深拷贝 1. 浅拷贝 浅拷贝是指只拷贝对象的值类型成员,而

C#中的浅拷贝与深拷贝

## 前言 众所周知,C#中有两种类型变量:那就是**值类型**和**引用类型**。对于值类型而言,copy就相当于是全盘复制了,真正的实现了复制,属于**深拷贝**;而对于引用类型而言,一般的copy只是**浅拷贝**,只是copy到了引用对象的地址,相当于值传递了一个引用指针,==新的对象通过地

使用c#强大的表达式树实现对象的深克隆

一、表达式树的基本概念 表达式树是一个以树状结构表示的表达式,其中每个节点都代表表达式的一部分。例如,一个算术表达式 a + b 可以被表示为一个树,其中根节点是加法运算符,它的两个子节点分别是 a 和 b。在 LINQ(语言集成查询)中,表达式树使得能够将 C# 中的查询转换成其他形式的查询,比如

使用c#强大的表达式树实现对象的深克隆之解决循环引用的问题

在上一期博客里,我们提到使用使用c#强大的表达式树实现对象的深克隆,文章地址:https://www.cnblogs.com/gmmy/p/18186750。但是文章里没有解决如何实现循环引用的问题。 循环引用 在C#中,循环引用通常发生在两个或更多的对象相互持有对方的引用,从而形成一个闭环。这种情

深度解读《深度探索C++对象模型》之数据成员的存取效率分析(二)

C++对象在经过类的封装后,存取对象中的数据成员的效率是否相比C语言的结构体访问效率要低下?本篇将从C++类的不同定义形式来一一分析C++对象的数据成员的访问在编译器中是如何实现的,以及它们的存取效率如何?

从优秀源码中学到的两个技巧

设计一个不能被using的对象 在实际开发中为了避免命名空间污染,一般都不会using namespace std。但是如果一个对象写起来比较复杂,用using能大幅度地简化操作。现在假设我们要设计一个函数,它在一个作用域里面,使用它只能以A::B::C()这种形式。思考一下,如果我们放在命名空间下

C#.Net筑基-集合知识全解

.Net 中提供了一系列的管理对象集合的类型,数组、可变列表、字典等。从类型安全上集合分为两类,泛型集合 和 非泛型集合,传统的非泛型集合存储为Object,需要类型转。而泛型集合提供了更好的性能、编译时类型安全,推荐使用。

Json 基于类 Newtonsoft.Json.Linq.JToken 的应用简介【C# 基础】

在日常开发中,对于 Json 的使用还是比较频繁的,特别是 Json 对象和字符串或者实体对象之间的转换。今天一起来看下吧。

【OpenVINO™】使用OpenVINO™ C# API 部署 YOLO-World实现实时开放词汇对象检测

YOLO-World是一个融合了实时目标检测与增强现实(AR)技术的创新平台,旨在将现实世界与数字世界无缝对接。该平台以YOLO(You Only Look Once)算法为核心,实现了对视频中物体的快速准确识别,并通过AR技术将虚拟元素与真实场景相结合,为用户带来沉浸式的交互体验。在本文中,我们将...

简易的工厂设计模式

工厂设计模式是一种创建型设计模式,它提供了一种创建对象的最佳方式,而无需暴露对象的创建逻辑。在工厂模式中,我们定义一个接口或抽象类,该接口或抽象类用于创建对象,但让子类决定要实例化的类。工厂方法模式使类的实例化延迟到其子类。 下面是一个完整的C#实现案例: 首先,我们定义一个接口,用于创建对象: p