一区二区三区在线-一区二区三区亚洲视频-一区二区三区亚洲-一区二区三区午夜-一区二区三区四区在线视频-一区二区三区四区在线免费观看

服務(wù)器之家:專注于服務(wù)器技術(shù)及軟件下載分享
分類導(dǎo)航

PHP教程|ASP.NET教程|Java教程|ASP教程|編程技術(shù)|正則表達(dá)式|C/C++|IOS|C#|Swift|Android|VB|R語言|JavaScript|易語言|vb.net|

服務(wù)器之家 - 編程語言 - C# - C#簡單實(shí)現(xiàn)表達(dá)式目錄樹(Expression)

C#簡單實(shí)現(xiàn)表達(dá)式目錄樹(Expression)

2022-02-12 16:04如此拉風(fēng)的女人 C#

表達(dá)式目錄樹以數(shù)據(jù)形式表示語言級別代碼。數(shù)據(jù)存儲在樹形結(jié)構(gòu)中。表達(dá)式目錄樹中的每個節(jié)點(diǎn)都表示一個表達(dá)式。這篇文章給大家介紹C#簡單實(shí)現(xiàn)表達(dá)式目錄樹(Expression),需要的朋友參考下吧

1.什么是表達(dá)式目錄樹 :簡單的說是一種語法樹,或者說是一種數(shù)據(jù)結(jié)構(gòu)(Expression)

2.用Lambda聲明表達(dá)式目錄樹: 

?
1
2
3
4
5
Expression<Func<int, int, int>> exp = (n, m) => n * m + 2; //表達(dá)試目錄樹的方法體只能是一行,不能有大括號。比如:
 //Expression<Func<int, int, int>> exp1 = (m, n) =>
 // {
 // return m * n + 2;
 // };

 3.Expression.Compile();

?
1
2
3
4
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的結(jié)果一樣,但是能Compile()的只有LambdaExpression。 Compile() 是將表達(dá)式樹描述的 Lambda 表達(dá)式編譯為可執(zhí)行代碼,并生成表示該 lambda 表達(dá)式的委托。exp.Compile().Invoke(99,99) 相當(dāng)于這樣調(diào)用 exp.Compile()();

4.認(rèn)識表達(dá)式目錄樹結(jié)構(gòu)。把上面的表達(dá)式拆分就是如下圖,小學(xué)數(shù)學(xué)知識里的,按照運(yùn)算符優(yōu)先級別,先算乘法,m*n,得出結(jié)果再算加法,加上2。

C#簡單實(shí)現(xiàn)表達(dá)式目錄樹(Expression)

如代碼所示,m和n是參數(shù),所以類型為ParameterExpression ,2是常量,常量類型是ConstantExpression ,MultiplyAssign 乘法,Add加法。第六步中只能執(zhí)行表示Lambda表達(dá)式的表達(dá)式目錄樹,即LambdaExpression或者Expression<TDelegate>類型。如果表達(dá)式目錄樹不是表示Lambda表達(dá)式,需要調(diào)用Lambda方法創(chuàng)建一個新的表達(dá)式。actExpression.Compile()成委托,再調(diào)用。

?
1
2
3
4
5
6
7
8
9
10
{
    ParameterExpression left = Expression.Parameter(typeof(int), "m");//左邊的參數(shù)
    ParameterExpression right = Expression.Parameter(typeof(int), "n");//右邊的參數(shù)
    ConstantExpression constantlExp = Expression.Constant(2,typeof(int));//常量2
    BinaryExpression binaryExpMult = Expression.MultiplyAssign(left, right);//兩個參數(shù)相乘
    BinaryExpression binaryExpAdd=Expression.Add(binaryExpMult, constantlExp);//相乘的結(jié)果再加2
    Expression<Func<int, int,int>> actExpression = Expression.Lambda<Func<int, int, int>>(binaryExpAdd, left, right);
    int result= actExpression.Compile()(2, 1);//調(diào)用
    Console.WriteLine(result+"");
}

 一些表達(dá)式目錄樹常用的類型

C#簡單實(shí)現(xiàn)表達(dá)式目錄樹(Expression)

 5.表達(dá)式目錄樹+緩存

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
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; }
 }
}

有時候一些業(yè)務(wù)模型和實(shí)體模型不太一樣,比如Student 于StudentDto實(shí)體的轉(zhuǎn)換

一般的寫法,new 一個實(shí)體然后把值賦給另一個實(shí)體,有一個就寫一個,有十個就寫是個,代碼寫死了,硬編碼性能高

?
1
2
3
4
5
6
7
8
9
{
    Student student = new Student() { Age = 12, Id=1, Name="晴天" };
    StudentDto studentDto = new StudentDto()
    {
     Name = student.Name,
     Id = student.Id,
     Age = student.Age
    };
   }

第二種:使用Expression表達(dá)式目錄樹

?
1
2
3
4
5
6
7
Expression<Func<Student, StudentDto>> lambda = p => new StudentDto
     {
      Age = p.Age,
      Id = p.Id,
      Name = p.Name
     };
     lambda.Compile().Invoke(student);

01.使用字典緩存表達(dá)式樹,第一步是實(shí)例化了一個命令參數(shù),parameterExpression,  List<MemberBinding> memberBindingList = new List<MemberBinding>();是一個對象成員集合列表,循環(huán)TOut的所有公共的屬性和字段,Add到memberBindingList集合中,然后使用MemberInitExpression初始化多個對象拼裝再調(diào)用。第一次調(diào)用動態(tài)拼裝,組裝了一個key放入字典中,緩存之后,就直接調(diào)用字典中的數(shù)據(jù)。緩存后的就是硬編碼所以性能高。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
using System;
using System.Collections.Generic;
using System.Linq;
using System.Linq.Expressions;
using System.Text;
using System.Threading.Tasks;
 
namespace ThreeHomeWork.MappingExtend
{
 /// <summary>
 /// 生成表達(dá)式目錄樹。字典緩存
 /// </summary>
 public class ExpressionMapper
 {
  private static Dictionary<string, object> _DIC = new Dictionary<string, object>();
 
  /// <summary>
  /// 字典緩存表達(dá)式樹
  /// </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類型的反射,通過反射遍歷賦值。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
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>();//創(chuàng)建對象
   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是比較好的一個工具,這種方式序列化代碼雖然一行搞定,但是序列化和反序列化的動作比反射動作大點(diǎn),耗時會比較高。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
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.生成表達(dá)式目錄樹,泛型緩存,使用泛型緩存性能是最高的。動態(tài)實(shí)現(xiàn)Student與StudentDto的轉(zhuǎn)換。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
using System;
using System.Collections.Generic;
using System.Linq;
using System.Linq.Expressions;
using System.Text;
using System.Threading.Tasks;
namespace ThreeHomeWork.MappingExtend
{
 /// <summary>
 /// 生成表達(dá)式目錄樹 泛型緩存
 /// </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);
  }
 }
}

總結(jié)

以上所述是小編給大家介紹的C#簡單實(shí)現(xiàn)表達(dá)式目錄樹(Expression),希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復(fù)大家的。在此也非常感謝大家對服務(wù)器之家網(wǎng)站的支持!

原文鏈接:http://www.cnblogs.com/wwym/p/7857022.html?utm_source=tuicool&utm_medium=referral

延伸 · 閱讀

精彩推薦
主站蜘蛛池模板: 大陆日韩欧美 | 免费一区 | 国产91页 | 好吊色永久免费视频大全 | 色综七七久久成人影 | 99视频免费在线观看 | 欧美巨吊 | 99热这里只有精 | 国产精品久久久久久影视 | 91制片厂制作果冻传媒破解 | 好妈妈7在线观看高清 | 亚洲香蕉网久久综合影院3p | 精品综合一区二区三区 | 亚洲 日韩经典 中文字幕 | 国产91区 | 国产精品国产精品国产三级普 | 国产日产精品久久久久快鸭 | 亚洲视频一区在线播放 | 色姑娘久久 | 性夜影院爽黄A爽免费动漫 性色欲情网站IWWW九文堂 | 欧美作爱福利免费观看视频 | 狠狠干奇米 | fuqer日本 | 午夜理论电影在线观看亚洲 | 国产东北3p真实在线456视频 | 肉色欧美久久久久久久蜜桃 | 久久噜国产精品拍拍拍拍 | 99热自拍 | 九九热在线观看视频 | 国产亚洲综合精品一区二区三区 | 无码人妻精品一区二区蜜桃在线看 | 男女男精品网站 | 久久亚洲精品AV成人无 | 日日插插| 精品免费久久久久久成人影院 | 男人把j放进女人的p里视频 | 亚洲AV人无码综合在线观看蜜桃 | 91成人免费视频 | 国产欧美一区二区三区免费看 | 国产小嫩模好紧 | 好大好猛好爽好深视频免费 |