
最近在写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表达式类型的委托。