5.3表达式树

简介: 表达式树对应`Expression<TDelegate>`类型,从Lambda表达式生成表达式树

5.3表达式树

表达式树对应Expression<TDelegate>类型,从Lambda表达式生成表达式树:

Expression<Func<Book,bool>> e=b=>b.Price>5;可以看到,编译器自动将Lambda表达式编译为Expression类型

Expression和Func的区别:Expression中存储了运算逻辑,而Func没有。简单来讲,Expression类似于代码,而Func类似于编译后的程序。

Func<Book, bool>f1=b=>b.Price>5||b.AuthorName.Contains("aaa");

Expression<Func<Book, bool>>e=b=>b.Price>5||b.AuthorName.Contains("aaa");

Console.WriteLine(f1);

Console.WriteLine(e);

可以看出,Func输出结果中,只有参数和返回值,内部没有运算逻辑。而Expression中存储了运算逻辑,要使用EF Core必须使用带运算逻辑的Expression。

查看表达式树

  • 可以在调试的时候通过快速监视来查看表达式树
  • 可以NuGet安装ExpressionTreeToString,调用ToString扩展方法来输出表达式树结构的字符串

Console.WriteLine(e.ToString("Object notation", "C#"));

通过代码动态创建表达式树

构造如下表达式树Expression<Func<Book,bool>> e = b=>b.Pricce>5;

//创建b参数节点

ParameterExpressionparamB=Expression.Parameter(typeof(Book), "b");

//创建访问b的属性操作节点

MemberExpressionexprLeft=Expression.MakeMemberAccess(paramB, typeof(Book).GetProperty("Price"));

//创建对应5这个常量的节点,要写成5.0因为是double类型对应数据库表中的类型,要格外注意

ConstantExpressionexprRight=Expression.Constant(5.0, typeof(double));

//创建对应大于符号的二元运算符节点,并把exprLeft和exprRight分别设定为左右节点

BinaryExpressionexprBody=Expression.MakeBinary(ExpressionType.GreaterThan, exprLeft, exprRight);

//使用Lambda方法把exprBody放到一个表达式树节点中

Expression<Func<Book, bool>>expr1=Expression.Lambda<Func<Book, bool>>(exprBody, paramB);

ctx.Books.Where(expr1).ToList();

Console.WriteLine(expr1.ToString("Object notation", "C#"));

如上创建表达式树太过复杂,可以使用ExpressionTreeToString提供的ToString("Factory methods", "C#")直接输出代码,然后使用using static方法引入Expression类,进行简单的改写就可以生成,注意常量类型,要使用5.0形式。另外,对于double和int等基本类型相等的比较和对string等复杂类型的比较不一样,对于基本类型,调用的是Equal方法,对于复杂类型,则要调用==重载运算符的重载方法op_Equality。

//动态创建Expression表达式树,并且可传递参数

IEnumerable<Book>QueryBooks(stringpropName, objectvalue)

{

   Typetype=typeof(Book);

   PropertyInfopropInfo=type.GetProperty(propName);

   TypepropType=propInfo.PropertyType;

   varb=Parameter(typeof(Book),"b");

   Expression<Func<Book,bool>>expr;

   if (propType.IsPrimitive)//如果是int、double等基本数据类型

   {

       expr=Lambda<Func<Book, bool>>(Equal(

               MakeMemberAccess(b,typeof(Book).GetProperty(propName)),

               Constant(value)),b);

   }

   else//如果是string等类型

   {

       expr=Lambda<Func<Book, bool>>(MakeBinary(ExpressionType.Equal,

               MakeMemberAccess(b,typeof(Book).GetProperty(propName)),

               Constant(value), false,propType.GetMethod("op_Equality")

           ),b);

   }

   TestDbContextctx=newTestDbContext();

   returnctx.Books.Where(expr).ToArray();

}

实现 Select的动态化

实现Select(b=>new{b.Id,b.Name})这种匿名类的动态化

usingSystem.Linq.Expressions;

IEnumerable<object[]>Query<TEntity>(string[] propNames) whereTEntity : class

{

   //创建参数节点

   ParameterExpressionexParameter=Expression.Parameter(typeof(TEntity));

   List<Expression>exProps=newList<Expression>();//创建表达式集合

   foreach (stringpropNameinpropNames)

   {

       ExpressionexProp=Expression.Convert(Expression.MakeMemberAccess(

           exParameter,typeof(TEntity).GetProperty(propName)), typeof(object));

       exProps.Add(exProp);

   }

   Expression[] initializers=exProps.ToArray();//创建表达式数组

   //NewArrayExpression代表数组

   NewArrayExpressionnewArrayExp=Expression.NewArrayInit(typeof(object), initializers);

   //使用Lambda方法把NewArrayExpression放到一个表达式树节点中

   varselectExpression=Expression.Lambda<Func<TEntity, object[]>>(newArrayExp, exParameter);

   usingTestDbContextctx=newTestDbContext();

   IQueryable<object[]>selectQueryable=ctx.Set<TEntity>().Select(selectExpression);

   returnselectQueryable.ToArray();

}

varitems=Query<Book>(newstring[] { "Id", "PubTime", "Title" });

foreach (object[] rowinitems)

{

   longid= (long)row[0];

   DateTimepubTime= (DateTime)row[1];

   stringtitle= (string)row[2];

   Console.WriteLine(id+","+pubTime+","+title);

}

相关文章
|
4月前
|
设计模式 安全 C++
奇异递归模板模式
CRTP(Curiously Recurring Template Pattern,奇异递归模板模式)是一种通过模板类与继承结合实现静态多态性的设计模式。其核心思想是让一个类通过模板参数继承自一个模板类,且该模板参数即为类本身。CRTP能够实现静态多态性,提高运行效率,同时提供代码复用、类型安全及避免菱形继承问题等优点。在C++示例中,`Base`模板基类提供通用功能,`Derived1`和`Derived2`分别继承自`Base&lt;Derived1&gt;`和`Base&lt;Derived2&gt;`,并实现各自的具体功能。
|
7月前
|
分布式计算 算法 Python
Python函数进阶:四大高阶函数、匿名函数、枚举、拉链与递归详解
Python函数进阶:四大高阶函数、匿名函数、枚举、拉链与递归详解
|
8月前
|
编译器 C++ 索引
C learning_13 操作符前篇(条件操作符、 逗号表达式、 下标引用、函数调用和结构成员、 表达式求值)
C learning_13 操作符前篇(条件操作符、 逗号表达式、 下标引用、函数调用和结构成员、 表达式求值)
|
8月前
|
SQL 关系型数据库 MySQL
MySQL数据类型、运算符以及数据库范式
MySQL数据类型、运算符以及数据库范式
82 0
MySQL数据类型、运算符以及数据库范式
|
8月前
|
NoSQL 容器 消息中间件
栈和队列及表达式求值问题
栈和队列及表达式求值问题
|
存储 数据挖掘 测试技术
抽象数据类型
抽象数据类型(Abstract Data Type,简称 ADT)是一种用于描述数据结构和操作的数据类型,它将数据对象、数据对象之间的关系和数据对象的基本操作封装在一起。ADT 可以用于任何需要处理数据和操作数据的地方,它的主要目的是提高代码的可重用性、模块化和抽象化。
114 1
|
存储 测试技术 C#
C#编程流程控制与集合类型(一)
C#编程流程控制与集合类型
C#编程流程控制与集合类型(一)
|
存储 编译器 C#
|
存储 C语言 索引
【C】操作符归纳+表达式运算原理
【C】操作符归纳+表达式运算原理
241 0
【C】操作符归纳+表达式运算原理
|
算法 Java
数据结构—栈的应用举例【算术表达式求值转换、后缀表达式求值计算】
数据结构—栈的应用举例【算术表达式求值转换、后缀表达式求值计算】
205 0
数据结构—栈的应用举例【算术表达式求值转换、后缀表达式求值计算】

热门文章

最新文章