C#简单实现表达式目录树(Expression)

 更新时间:2020年6月25日 11:18  点击:2198

1.什么是表达式目录树 :简单的说是一种语法树,或者说是一种数据结构(Expression)

2.用Lambda声明表达式目录树: 

Expression<Func<int, int, int>> exp = (n, m) => n * m + 2; //表达试目录树的方法体只能是一行,不能有大括号。比如:
 //Expression<Func<int, int, int>> exp1 = (m, n) =>
 // {
 // return m * n + 2;
 // };

 3.Expression.Compile();

 Func<int, int, int> func = (m, n) => m * n + 2;
 Expression<Func<int, int, int>> exp = (m, n) => m * n + 2;
 int iResult1 = func.Invoke(99, 99);
 int iResult2 = exp.Compile().Invoke(99, 99);

iResult1 和iResult2的结果一样,但是能Compile()的只有LambdaExpression。 Compile() 是将表达式树描述的 Lambda 表达式编译为可执行代码,并生成表示该 lambda 表达式的委托。exp.Compile().Invoke(99,99) 相当于这样调用 exp.Compile()();

4.認識表达式目录树结构。把上面的表达式拆分就是如下图,小学数学知识里的,按照运算符优先级别,先算乘法,m*n,得出结果再算加法,加上2。

如代码所示,m和n是参数,所以类型为ParameterExpression ,2是常量,常量类型是ConstantExpression ,MultiplyAssign 乘法,Add加法。第六步中只能执行表示Lambda表达式的表达式目录树,即LambdaExpression或者Expression<TDelegate>类型。如果表达式目录树不是表示Lambda表达式,需要调用Lambda方法创建一个新的表达式。actExpression.Compile()成委托,再调用。

 {
     ParameterExpression left = Expression.Parameter(typeof(int), "m");//左边的参数
     ParameterExpression right = Expression.Parameter(typeof(int), "n");//右边的参数
     ConstantExpression constantlExp = Expression.Constant(2,typeof(int));//常量2
     BinaryExpression binaryExpMult = Expression.MultiplyAssign(left, right);//两个参数相乘
     BinaryExpression binaryExpAdd=Expression.Add(binaryExpMult, constantlExp);//相乘的结果再加2
     Expression<Func<int, int,int>> actExpression = Expression.Lambda<Func<int, int, int>>(binaryExpAdd, left, right);
     int result= actExpression.Compile()(2, 1);//调用
     Console.WriteLine(result+"");
 }

 一些表达式目录树常用的类型

 5.表达式目录树+缓存

 using System;
 using System.Collections.Generic;
 using System.Linq;
 using System.Text;
 using System.Threading.Tasks;
 namespace ThreeHomeWork.Model
 {
  public class Student
  {
   public int Id { get; set; }
   public string Name { get; set; }
   public int Age { get; set; }
  }
  public class StudentDto
  {
   public int Id { get; set; }
   public string Name { get; set; }
   public int Age { get; set; }
  }
 }

有时候一些业务模型和实体模型不太一样,比如Student 于StudentDto实体的转换

一般的写法,new 一个实体然后把值赋给另一个实体,有一个就写一个,有十个就写是个,代码写死了,硬编码性能高

 {
     Student student = new Student() { Age = 12, Id=1, Name="晴天" };
     StudentDto studentDto = new StudentDto()
     {
      Name = student.Name,
      Id = student.Id,
      Age = student.Age
     };
    }

第二种:使用Expression表达式目录树

Expression<Func<Student, StudentDto>> lambda = p => new StudentDto
     {
      Age = p.Age,
      Id = p.Id,
      Name = p.Name
     };
     lambda.Compile().Invoke(student);

01.使用字典缓存表达式树,第一步是实例化了一个命令参数,parameterExpression,  List<MemberBinding> memberBindingList = new List<MemberBinding>();是一个对象成员集合列表,循环TOut的所有公共的属性和字段,Add到memberBindingList集合中,然后使用MemberInitExpression初始化多个对象拼装再调用。第一次调用动态拼装,组装了一个key放入字典中,缓存之后,就直接调用字典中的数据。缓存后的就是硬编码所以性能高。

 using System;
 using System.Collections.Generic;
 using System.Linq;
 using System.Linq.Expressions;
 using System.Text;
 using System.Threading.Tasks;
 
 namespace ThreeHomeWork.MappingExtend
 {
  /// <summary>
  /// 生成表达式目录树。字典缓存
  /// </summary>
  public class ExpressionMapper
  {
   private static Dictionary<string, object> _DIC = new Dictionary<string, object>();
 
   /// <summary>
   /// 字典缓存表达式树
   /// </summary>
   /// <typeparam name="TIn"></typeparam>
   /// <typeparam name="TOut"></typeparam>
   /// <param name="tIn"></param>
   /// <returns></returns>
   public static TOut Trans<TIn, TOut>(TIn tIn)
   {
    string key = string.Format("funckey_{0}_{1}", typeof(TIn).FullName, typeof(TOut).FullName);
    if (!_DIC.ContainsKey(key))
    {
     ParameterExpression parameterExpression = Expression.Parameter(typeof(TIn), "p");
     List<MemberBinding> memberBindingList = new List<MemberBinding>();
     foreach (var item in typeof(TOut).GetProperties())
     {
      MemberExpression property = Expression.Property(parameterExpression, typeof(TIn).GetProperty(item.Name));
      MemberBinding memberBinding = Expression.Bind(item, property);
      memberBindingList.Add(memberBinding);
     }
     foreach (var item in typeof(TOut).GetFields())
     {
      MemberExpression property = Expression.Field(parameterExpression, typeof(TIn).GetField(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
     });
     Func<TIn, TOut> func = lambda.Compile();//拼装是一次性的
     _DIC[key] = func;
    }
    return ((Func<TIn, TOut>)_DIC[key]).Invoke(tIn);
   }
 
  }
 }

02.泛型+反射,接收一个TIn类型的,返回一个TOut类型的反射,通过反射遍历赋值。

 using System;
 using System.Collections.Generic;
 using System.Linq;
 using System.Text;
 using System.Threading.Tasks;
 namespace ThreeHomeWork.MappingExtend
 {
  public class ReflectionMapper
  {
   /// <summary>
   /// 反射
   /// </summary>
   /// <typeparam name="TIn"></typeparam>
   /// <typeparam name="TOut"></typeparam>
   /// <param name="tIn"></param>
   /// <returns></returns>
   public static TOut Trans<TIn, TOut>(TIn tIn)
   {
    TOut tOut = Activator.CreateInstance<TOut>();//创建对象
    foreach (var itemOut in tOut.GetType().GetProperties())//遍历属性
    {
     foreach (var itemIn in tIn.GetType().GetProperties())
     {
      if (itemOut.Name.Equals(itemIn.Name))
      {
       itemOut.SetValue(tOut, itemIn.GetValue(tIn));
       break;
      }
     }
    }
    foreach (var itemOut in tOut.GetType().GetFields())//遍历字段
    {
     foreach (var itemIn in tIn.GetType().GetFields())
     {
      if (itemOut.Name.Equals(itemIn.Name))
      {
       itemOut.SetValue(tOut, itemIn.GetValue(tIn));
       break;
      }
     }
    }
    return tOut;
   }
  }
 }

03.使用第三方序列化反序列化工具,Newtonsoft.Json是比较好的一个工具,这种方式序列化代码虽然一行搞定,但是序列化和反序列化的动作比反射动作大点,耗时会比较高。

 using Newtonsoft.Json;
 using System;
 using System.Collections.Generic;
 using System.Linq;
 using System.Text;
 using System.Threading.Tasks;
 
 namespace ExpressionDemo.MappingExtend
 {
  public class SerializeMapper
  {
   /// <summary>
   /// 序列化反序列化方式
   /// </summary>
   /// <typeparam name="TIn"></typeparam>
   /// <typeparam name="TOut"></typeparam>
   public static TOut Trans<TIn, TOut>(TIn tIn)
   {
    return JsonConvert.DeserializeObject<TOut>(JsonConvert.SerializeObject(tIn));
   }
  }
 }

04.生成表达式目录树,泛型缓存,使用泛型缓存性能是最高的。动态实现Student与StudentDto的转换。

 using System;
 using System.Collections.Generic;
 using System.Linq;
 using System.Linq.Expressions;
 using System.Text;
 using System.Threading.Tasks;
 namespace ThreeHomeWork.MappingExtend
 {
  /// <summary>
  /// 生成表达式目录树 泛型缓存
  /// </summary>
  /// <typeparam name="TIn"></typeparam>
  /// <typeparam name="TOut"></typeparam>
  public class ExpressionGenericMapper<TIn, TOut>//Mapper`2
  {
   private static Func<TIn, TOut> _FUNC = null;
   static ExpressionGenericMapper()
   {
    ParameterExpression parameterExpression = Expression.Parameter(typeof(TIn), "p");
    List<MemberBinding> memberBindingList = new List<MemberBinding>();
    foreach (var item in typeof(TOut).GetProperties())
    {
     MemberExpression property = Expression.Property(parameterExpression, typeof(TIn).GetProperty(item.Name));
     MemberBinding memberBinding = Expression.Bind(item, property);
     memberBindingList.Add(memberBinding);
    }
    foreach (var item in typeof(TOut).GetFields())
    {
     MemberExpression property = Expression.Field(parameterExpression, typeof(TIn).GetField(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
    });
    _FUNC = lambda.Compile();//拼装是一次性的
   }
   public static TOut Trans(TIn t)
   {
    return _FUNC(t);
   }
  }
 }

总结

以上所述是小编给大家介绍的C#简单实现表达式目录树(Expression),希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对猪先飞网站的支持!

[!--infotagslink--]

相关文章

  • C#实现简单的登录界面

    我们在使用C#做项目的时候,基本上都需要制作登录界面,那么今天我们就来一步步看看,如果简单的实现登录界面呢,本文给出2个例子,由简入难,希望大家能够喜欢。...2020-06-25
  • 浅谈C# 字段和属性

    这篇文章主要介绍了C# 字段和属性的的相关资料,文中示例代码非常详细,供大家参考和学习,感兴趣的朋友可以了解下...2020-11-03
  • C#中截取字符串的的基本方法详解

    这篇文章主要介绍了C#中截取字符串的的基本方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2020-11-03
  • C#实现简单的Http请求实例

    这篇文章主要介绍了C#实现简单的Http请求的方法,以实例形式较为详细的分析了C#实现Http请求的具体方法,需要的朋友可以参考下...2020-06-25
  • C#连接SQL数据库和查询数据功能的操作技巧

    本文给大家分享C#连接SQL数据库和查询数据功能的操作技巧,本文通过图文并茂的形式给大家介绍的非常详细,需要的朋友参考下吧...2021-05-17
  • C#中new的几种用法详解

    本文主要介绍了C#中new的几种用法,具有很好的参考价值,下面跟着小编一起来看下吧...2020-06-25
  • 使用Visual Studio2019创建C#项目(窗体应用程序、控制台应用程序、Web应用程序)

    这篇文章主要介绍了使用Visual Studio2019创建C#项目(窗体应用程序、控制台应用程序、Web应用程序),小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧...2020-06-25
  • C#开发Windows窗体应用程序的简单操作步骤

    这篇文章主要介绍了C#开发Windows窗体应用程序的简单操作步骤,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2021-04-12
  • C#从数据库读取图片并保存的两种方法

    这篇文章主要介绍了C#从数据库读取图片并保存的方法,帮助大家更好的理解和使用c#,感兴趣的朋友可以了解下...2021-01-16
  • C#和JavaScript实现交互的方法

    最近做一个小项目不可避免的需要前端脚本与后台进行交互。由于是在asp.net中实现,故问题演化成asp.net中jiavascript与后台c#如何进行交互。...2020-06-25
  • C++调用C#的DLL程序实现方法

    本文通过例子,讲述了C++调用C#的DLL程序的方法,作出了以下总结,下面就让我们一起来学习吧。...2020-06-25
  • 轻松学习C#的基础入门

    轻松学习C#的基础入门,了解C#最基本的知识点,C#是一种简洁的,类型安全的一种完全面向对象的开发语言,是Microsoft专门基于.NET Framework平台开发的而量身定做的高级程序设计语言,需要的朋友可以参考下...2020-06-25
  • C#变量命名规则小结

    本文主要介绍了C#变量命名规则小结,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...2021-09-09
  • C#绘制曲线图的方法

    这篇文章主要介绍了C#绘制曲线图的方法,以完整实例形式较为详细的分析了C#进行曲线绘制的具体步骤与相关技巧,具有一定参考借鉴价值,需要的朋友可以参考下...2020-06-25
  • C# 中如何取绝对值函数

    本文主要介绍了C# 中取绝对值的函数。具有很好的参考价值。下面跟着小编一起来看下吧...2020-06-25
  • c#自带缓存使用方法 c#移除清理缓存

    这篇文章主要介绍了c#自带缓存使用方法,包括获取数据缓存、设置数据缓存、移除指定数据缓存等方法,需要的朋友可以参考下...2020-06-25
  • c#中(&&,||)与(&,|)的区别详解

    这篇文章主要介绍了c#中(&&,||)与(&,|)的区别详解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧...2020-06-25
  • 经典实例讲解C#递归算法

    这篇文章主要用实例讲解C#递归算法的概念以及用法,文中代码非常详细,帮助大家更好的参考和学习,感兴趣的朋友可以了解下...2020-06-25
  • C#学习笔记- 随机函数Random()的用法详解

    下面小编就为大家带来一篇C#学习笔记- 随机函数Random()的用法详解。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧...2020-06-25
  • C#中list用法实例

    这篇文章主要介绍了C#中list用法,结合实例形式分析了C#中list排序、运算、转换等常见操作技巧,具有一定参考借鉴价值,需要的朋友可以参考下...2020-06-25