C#浅拷贝和深拷贝数据

目录

一、浅拷贝

二、深拷贝


一、浅拷贝

就是把原来的数据,复制一份,但是2份数据共享地址的,修改第一份数据或者修改第二份数据,都会一起改变,这可能不是我们程序中需要的场景。

下面我们演示一下,首先建立一个树结构的数据类型,其他类型也可以

1.建立项目

2.建立树结构数据

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace WpfApp4
{
    public class TestData1
    {
        public int ID { get; set; }
        public string D1 { get; set; }
        public string D2 { get; set; }

        public TestData2 testData2 { get; set; }
    }

    public class TestData2
    {
        public int ID { get; set; }
        public string D1 { get; set; }
        public string D2 { get; set; }

        public TestData3 testData3 { get; set; }
    }

    public class TestData3
    {
        public int ID { get; set; }
        public string D1 { get; set; }
        public string D2 { get; set; }
    }
}

2.效果

此时testData1的值,呈现树结构显示

把testData1的值赋值给test1后,依然呈现树结构显示

当修改testData1的D1的值后,test1的D1的值也改变了

这个就是浅拷贝,这种拷贝在程序场景中很少用到,不知道的人,偶尔会很奇怪。

二、深拷贝

就是把原来的数据,复制一份,但是2份数据不是共享地址的,修改第一份数据或者修改第二份数据,不会一起改变,这种场景大部分是我们程序中需要的场景。

前面的数据结构依然不变,深拷贝在程序中运用的场景非常多,我们说4种方法。

1.反射实现

可见,修改testData1的D1的值后,test2的D1的值,并没有变化,这正是我们需要的,后续可以对testData1和test2的数据,分别进行业务操作。 

2.JSON字符串序列化

此处需要使用Newtonsoft.Json这个包

这里我们首先是先把对象转成字符串,再把字符串转成对象,效果和上面是一样的。 

补充:也可以使用Net6中自带的序列化,需要引用System.Text.Json

效果一样,代码如下:

           var test31 = JsonSerializer.Deserialize<TestData1>(JsonSerializer.Serialize(testData1));   //首先是先把对象转成字符串,再把字符串转成对象
            testData1.D1 = "我修改了";

3.表达式树

和上面的效果一样。

4.AutoMapper

这个是第三方的,也可以使用其他的对象映射

此时我们需要先安装

安装完成后,第一句代码,需要进行配置,然后直接调用即可 

可见,效果都是一样的。 

代码:


using AutoMapper;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Linq.Expressions;
using System.Reflection;
using System.Text;
using System.Text.Json;
using System.Threading.Tasks;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Navigation;
using System.Windows.Shapes;
using static WpfApp4.MainWindow;
using Expression = System.Linq.Expressions.Expression;

namespace WpfApp4
{
    /// <summary>
    /// Interaction logic for MainWindow.xaml
    /// </summary>
    public partial class MainWindow : Window
    {
        public MainWindow()
        {
            InitializeComponent();

            TestData3 testData3 = new TestData3();
            testData3.ID = 3;
            testData3.D1 = "3.1";
            testData3.D2 = "3.2";

            TestData2 testData2 = new TestData2();
            testData2.ID = 2;
            testData2.D1 = "2.1";
            testData2.D2 = "2.2";
            testData2.testData3 = testData3;

            TestData1 testData1 = new TestData1();
            testData1.ID = 1;
            testData1.D1 = "1.1";
            testData1.D2 = "1.2";
            testData1.testData2 = testData2;

            #region 浅拷贝演示
            浅拷贝演示,把testData1复制一份
            //TestData1 test1 = testData1;
            修改test1的值,那么testData1的值也会变化
            //testData1.D1 = "我修改了";
            #endregion

            #region 深拷贝演示
            //1.反射
            //var test2 = DeepCopyWithReflection(testData1);
            //testData1.D1 = "我修改了";
            //2.JSON字符串序列化
            //var test3 = JsonConvert.DeserializeObject<TestData1>(JsonConvert.SerializeObject(testData1));   //首先是先把对象转成字符串,再把字符串转成对象
            //testData1.D1 = "我修改了";

            //var test31 = JsonSerializer.Deserialize<TestData1>(JsonSerializer.Serialize(testData1));   //首先是先把对象转成字符串,再把字符串转成对象
            //testData1.D1 = "我修改了";
            //3.表达式树
            //var test4 = TransExp<TestData1, TestData1>.Trans(testData1);
            //testData1.D1 = "我修改了";

            //4.AutoMapper
            var config = new MapperConfiguration(cfg => cfg.CreateMap<TestData1, TestData1>());//映射配置
            var test5 = config.CreateMapper().Map<TestData1>(testData1);
            testData1.D1 = "我修改了";
            #endregion

        }



        /// <summary>
        /// 利用反射实现深拷贝
        /// </summary>
        /// <typeparam name="T"></typeparam>
        /// <param name="obj"></param>
        /// <returns></returns>
        public static T DeepCopyWithReflection<T>(T obj)
        {
            Type type = obj.GetType();

            // 如果是字符串或值类型则直接返回
            if (obj is string || type.IsValueType) return obj;
            // 如果是数组
            if (type.IsArray)
            {
                Type elementType = Type.GetType(type.FullName.Replace("[]", string.Empty));
                var array = obj as Array;
                Array copied = Array.CreateInstance(elementType, array.Length);
                for (int i = 0; i < array.Length; i++)
                {
                    copied.SetValue(DeepCopyWithReflection(array.GetValue(i)), i);
                }
                return (T)Convert.ChangeType(copied, obj.GetType());
            }

            object retval = Activator.CreateInstance(obj.GetType());

            PropertyInfo[] properties = obj.GetType().GetProperties(
                BindingFlags.Public | BindingFlags.NonPublic
                | BindingFlags.Instance | BindingFlags.Static);
            foreach (var property in properties)
            {
                var propertyValue = property.GetValue(obj, null);
                if (propertyValue == null)
                    continue;
                property.SetValue(retval, DeepCopyWithReflection(propertyValue), null);
            }

            return (T)retval;
        }
        /// <summary>
        /// 表达式树
        /// </summary>
        /// <typeparam name="TIn"></typeparam>
        /// <typeparam name="TOut"></typeparam>
        public static class TransExp<TIn, TOut>
        {
            private static readonly Func<TIn, TOut> cache = GetFunc();
            private static Func<TIn, TOut> GetFunc()
            {
                ParameterExpression parameterExpression = Expression.Parameter(typeof(TIn), "p");
                List<MemberBinding> memberBindingList = new List<MemberBinding>();

                foreach (var item in typeof(TOut).GetProperties())
                {
                    if (!item.CanWrite) continue;
                    MemberExpression property = Expression.Property(parameterExpression, typeof(TIn).GetProperty(item.Name));
                    MemberBinding memberBinding = Expression.Bind(item, property);
                    memberBindingList.Add(memberBinding);
                }

                MemberInitExpression memberInitExpression = Expression.MemberInit(Expression.New(typeof(TOut)), memberBindingList.ToArray());
                Expression<Func<TIn, TOut>> lambda = Expression.Lambda<Func<TIn, TOut>>(memberInitExpression, new ParameterExpression[] { parameterExpression });

                return lambda.Compile();
            }

            public static TOut Trans(TIn tIn)
            {
                return cache(tIn);
            }

        }

    }
}

本案例代码:

https://download.csdn.net/download/u012563853/88637534

来源:

C#浅拷贝和深拷贝数据-CSDN博客

  • 9
    点赞
  • 23
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

故里2130

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值