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),希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對億速云網站的支持!
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。