我的编程空间,编程开发者的网络收藏夹
学习永远不晚

C#使用表达式树怎么实现对象复制

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

C#使用表达式树怎么实现对象复制

本篇内容主要讲解“C#使用表达式树怎么实现对象复制”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“C#使用表达式树怎么实现对象复制”吧!

需求背景:对象复制性能优化;同时,在对象复制时,应跳过引用类型的null值复制,值类型支持值类型向可空类型的复制

using Common;using System;class Program{    static void Main(string[] args)    {        TestClassA classA = new TestClassA() { PropA = new TestClass() { Name = "cs1" }, PropB = "c1", PropC = 1 };        TestClassA classB = new TestClassA() { PropA = new TestClass() { Name = "cs2" }, PropB = "c2", PropC = 2 };        FastCopy.Copy(classA, classB, false);        Console.WriteLine(classB.PropA?.Name + ":" + classB.PropB + ":" + classB.PropC);        TestClassA classC = new TestClassA() { PropA = new TestClass() { Name = "cs1" } };        TestClassA classD = new TestClassA() { PropA = new TestClass() { Name = "cs2" }, PropB = "c2", PropC = 2 };        FastCopy.Copy(classC, classD, false);        Console.WriteLine(classD.PropA?.Name + ":" + classD.PropB + ":" + classD.PropC);    }}public class TestClassA{    public TestClass PropA { get; set; }    public string PropB { get; set; }    public int? PropC { get; set; }}public class TestClass{    public string Name { get; set; }}

输出:

C#使用表达式树怎么实现对象复制

百万次调用耗时:270-300ms

using System;using System.Collections.Concurrent;using System.Collections.Generic;using System.Linq;using System.Linq.Expressions;using System.Reflection;using static System.Linq.Expressions.Expression;namespace Common{    public static class FastCopy    {        static ConcurrentDictionary<string, object> copiers = new ConcurrentDictionary<string, object>();        /// <summary>        /// 复制两个对象同名属性值        /// </summary>        /// <typeparam name="S"></typeparam>        /// <typeparam name="T"></typeparam>        /// <param name="source">源对象</param>        /// <param name="target">目标对象</param>        /// <param name="copyNull">源对象属性值为null时,是否将值复制给目标对象</param>        public static void Copy<S, T>(S source, T target, bool copyNull = true)        {            string name = string.Format("{0}_{1}_{2}", typeof(S), typeof(T), copyNull);            object targetCopier;            if (!copiers.TryGetValue(name, out targetCopier))            {                Action<S, T> copier = CreateCopier<S, T>(copyNull);                copiers.TryAdd(name, copier);                targetCopier = copier;            }            Action<S, T> action = (Action<S, T>)targetCopier;            action(source, target);        }        /// <summary>        /// 为指定的两种类型编译生成属性复制委托        /// </summary>        /// <typeparam name="S"></typeparam>        /// <typeparam name="T"></typeparam>        /// <param name="copyNull">源对象属性值为null时,是否将值复制给目标对象</param>        /// <returns></returns>        private static Action<S, T> CreateCopier<S, T>(bool copyNull)        {            ParameterExpression source = Parameter(typeof(S));            ParameterExpression target = Parameter(typeof(T));            var sourceProps = typeof(S).GetProperties(BindingFlags.Instance | BindingFlags.Public).Where(p => p.CanRead).ToList();            var targetProps = typeof(T).GetProperties(BindingFlags.Instance | BindingFlags.Public).Where(p => p.CanWrite).ToList();            // 查找可进行赋值的属性            var copyProps = targetProps.Where(tProp => sourceProps.Where(sProp => sProp.Name == tProp.Name// 名称一致 且            && (            sProp.PropertyType == tProp.PropertyType// 属性类型一致 或            || sProp.PropertyType.IsAssignableFrom(tProp.PropertyType) // 源属性类型 为 目标属性类型 的 子类;eg:object target = string source;   或            || (tProp.PropertyType.IsValueType && sProp.PropertyType.IsValueType && // 属性为值类型且基础类型一致,但目标属性为可空类型 eg:int? num = int num;            ((tProp.PropertyType.GenericTypeArguments.Length > 0 ? tProp.PropertyType.GenericTypeArguments[0] : tProp.PropertyType) == sProp.PropertyType))            )).Count() > 0);            List<Expression> expressionList = new List<Expression>();            foreach (var prop in copyProps)            {                if (prop.PropertyType.IsValueType)// 属性为值类型                {                    PropertyInfo sProp = typeof(S).GetProperty(prop.Name);                    PropertyInfo tProp = typeof(T).GetProperty(prop.Name);                    if (sProp.PropertyType == tProp.PropertyType)// 属性类型一致 eg:int num = int num;    或   int? num = int? num;                    {                        var assign = Assign(Property(target, prop.Name), Property(source, prop.Name));                        expressionList.Add(assign);                    }                    else if (sProp.PropertyType.GenericTypeArguments.Length <= 0 && tProp.PropertyType.GenericTypeArguments.Length > 0)// 属性类型不一致且目标属性类型为可空类型 eg:int? num = int num;                    {                        var convert = Convert(Expression.Property(source, prop.Name), tProp.PropertyType);                        var cvAssign = Assign(Expression.Property(target, prop.Name), convert);                        expressionList.Add(cvAssign);                    }                }                else// 属性为引用类型                {                    var assign = Assign(Property(target, prop.Name), Property(source, prop.Name));// 编译生成属性赋值语句   target.{PropertyName} = source.{PropertyName};                    var sourcePropIsNull = Equal(Constant(null, prop.PropertyType), Property(source, prop.Name));// 判断源属性值是否为Null;编译生成  source.{PropertyName} == null                    var setNull = IsTrue(Constant(copyNull));// 判断是否复制Null值 编译生成  copyNull == True                    var setNullTest = IfThen(setNull, assign);                    var condition = IfThenElse(sourcePropIsNull, setNullTest, assign);                                        expressionList.Add(condition);                }            }            var block = Block(expressionList.ToArray());            Expression<Action<S, T>> lambda = Lambda<Action<S, T>>(block, source, target);            return lambda.Compile();        }    }}

如果完整复制,去掉逻辑判断,同时可通过泛型类,不在使用字典,性能还可以提升。

using System;using System.Linq;using System.Linq.Expressions;using System.Reflection;namespace Common{    public static class FastCopy<S, T>    {        static Action<S, T> action = CreateCopier();        /// <summary>        /// 复制两个对象同名属性值        /// </summary>        /// <typeparam name="S"></typeparam>        /// <typeparam name="T"></typeparam>        /// <param name="source">源对象</param>        /// <param name="target">目标对象</param>        /// <param name="copyNull">源对象属性值为null时,是否将值复制给目标对象</param>        public static void Copy(S source, T target, bool copyNull = true)        {            action(source, target);        }        /// <summary>        /// 为指定的两种类型编译生成属性复制委托        /// </summary>        /// <typeparam name="S"></typeparam>        /// <typeparam name="T"></typeparam>        /// <param name="copyNull">源对象属性值为null时,是否将值复制给目标对象</param>        /// <returns></returns>        private static Action<S, T> CreateCopier()        {            ParameterExpression source = Expression.Parameter(typeof(S));            ParameterExpression target = Expression.Parameter(typeof(T));            var sourceProps = typeof(S).GetProperties(BindingFlags.Instance | BindingFlags.Public).Where(p => p.CanRead).ToList();            var targetProps = typeof(T).GetProperties(BindingFlags.Instance | BindingFlags.Public).Where(p => p.CanWrite).ToList();            // 查找可进行赋值的属性            var copyProps = targetProps.Where(tProp => sourceProps.Where(sProp => sProp.Name == tProp.Name// 名称一致 且            && (            sProp.PropertyType == tProp.PropertyType// 属性类型一致            )).Count() > 0);            var block = Expression.Block(from p in copyProps select Expression.Assign(Expression.Property(target, p.Name), Expression.Property(source, p.Name)));            Expression<Action<S, T>> lambda = Expression.Lambda<Action<S, T>>(block, source, target);            return lambda.Compile();        }    }}

百万次耗时:100ms左右

到此,相信大家对“C#使用表达式树怎么实现对象复制”有了更深的了解,不妨来实际操作一番吧!这里是编程网网站,更多相关内容可以进入相关频道进行查询,关注我们,继续学习!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

C#使用表达式树怎么实现对象复制

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

C#使用表达式树怎么实现对象复制

本篇内容主要讲解“C#使用表达式树怎么实现对象复制”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“C#使用表达式树怎么实现对象复制”吧!需求背景:对象复制性能优化;同时,在对象复制时,应跳过引用类
2023-06-22

C# Lambda表达式树怎么实现

这篇文章主要介绍“C# Lambda表达式树怎么实现”,在日常操作中,相信很多人在C# Lambda表达式树怎么实现问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”C# Lambda表达式树怎么实现”的疑惑有所
2023-06-17

C++中怎么实现对象复制

今天就跟大家聊聊有关C++中怎么实现对象复制,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。C++对象复制代码示例:class Table{ Name * p; size_t sz
2023-06-17

Mybatis怎么使用ognl表达式实现动态sql

这篇文章主要为大家展示了“Mybatis怎么使用ognl表达式实现动态sql”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“Mybatis怎么使用ognl表达式实现动态sql”这篇文章吧。新建Us
2023-06-15

C#中怎么利用正则表达式实现预搜索

这篇文章将为大家详细讲解有关C#中怎么利用正则表达式实现预搜索,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。C#正则表达式之正向预搜索声明格式:正声明 “(?=…)”,负声明
2023-06-17

使用C语言怎么实现面向对象编程OOP

本篇文章为大家展示了使用C语言怎么实现面向对象编程OOP,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。过程&对象?一个对象就是由或多或少的针对这个对象的过程构成的,当然其中是少不了必要的属性。一个过
2023-06-16

怎么在python中使用正则表达式实现零宽断言

怎么在python中使用正则表达式实现零宽断言?相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。python的数据类型有哪些?python的数据类型:1. 数字类型,包括int(整
2023-06-14

Java中小数点后多余的0怎么使用正则表达式实现删除

这篇文章给大家介绍Java中小数点后多余的0怎么使用正则表达式实现删除,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。具体如下:package test;/** * 去掉多余的.与0 * @author Hust * @
2023-05-31

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录