最近在写ORM框架,其中遇到一个难点,就是作为框架调用方如何将查询条件传入框架内。其中就用到了Expression。
Func委托
要Expression先要了解Func委托,Func委托的样式是:
Func<T,TResult>
他是一种C#提供的固定的委托方法,算是微软提供的一种语法糖。
举例说明:
1 //声明 2 Func<int, int> func = f => f + 1; 3 //调用 4 int funcResult=func(1); 5 6 //结果=2
其中,Func<int, int>中第一个int是参数类型,第二个int是返回值类型。
Func<int, int> func = f => f + 1;等号后跟表达式中的参数,=>是lambda表达式的运算符,后跟方法体。
与下面这种正常的委托写法是一样的效果:
声明委托类型:
delegate int Function(int f);
赋值委托方法:
int ffunc(int f) { return f + 1; } //声明委托对象 Function function; //赋值对象 function = ffunc; int ffuncResult = function(1);
其中我们可以看出来,不管是上面Func的写法还是delegate写法,都是相当于把方法作为参数进行传递。
注意Func只能传递一个参数,但是可以自定义对象作为参数类型变相的传递多个参数。
Expression表达式树
理解了委托和上面的Func委托Expression就很容易理解了。
先看Expression的官方文档是怎么说的:
~~好吧,官方文档实在看不懂,那F12一下:
这里就能看出来了一部分内容,Expression是要使用委托类型的,而且只能是lambda表达式类型的委托,所以最好就是Func委托,这也是为什么刚开始我要讲Func委托的原因。
他的使用形式上是这样的:
Expression<Func<T,TResult>>
可以看出他是在委托上又嵌套了一层,就像树一样,“委托树”,因为只能嵌套lambda表达式,所以官方叫法是表达式树。
那要怎么用呢?此处列举一种用法,就是我开头提到的获取对象的属性名称。
声明实体:
public class Class2 { public string A1 { get; set; } public int? A2 { get; set; } }
获取属性名称方法:
//获取属性名称 string ConvertOrderBy<TEntity>(Expression<Func<TEntity, object>> orderby) where TEntity : class { var member = orderby.Body as MemberExpression; var unary = orderby.Body as UnaryExpression; return member != null ? member.Member.Name : (unary != null ? (unary.Operand as MemberExpression).Member.Name : null); } string SetOrderBy<TEntity>(Expression<Func<TEntity,object>> orderBy) where TEntity : class { return ConvertOrderBy(orderBy); } var linqStr = SetOrderBy<Class2>(s => s.A2);
运行结果:
总结
Expression在EF框架中存在大量使用,主要是需要框架底层需要知道传入属性的名称和类型。在使用时一定要注意Expression中一定只能是lambda表达式类型的委托。