7

我正在努力构建表达式树,以便可以动态地对某些数据进行过滤。

我想出了这个,但它失败var lambda =

foreach (var rule in request.Where.Rules)
{
    var parameterExpression = Expression.Parameter(typeof(string), rule.Field);
    var left = Expression.Call(parameterExpression, typeof(string).GetMethod("ToLower", Type.EmptyTypes));
    var right = Expression.Constant(rule.Data.ToLower());
    var method = typeof(string).GetMethod("Contains", new [] { typeof(string) });
    var call = Expression.Call(left, method, right);
    var lambda = Expression.Lambda<Func<T, bool>>(call, parameterExpression);
    query = query.Where(lambda);
}

varrule有一个字段(例如“名称”),我想将其与rule.Data(例如“tom”)中的文本进行比较。因此,如果T.Name.Contains("tom");我希望查询包含记录,否则不会。

varquery是类型IQueryable<T>

编辑:终于让它使用这个代码:

foreach (var rule in request.Where.Rules)
{
    var parameter = Expression.Parameter(typeof(T), "x");
    var property = Expression.Property(parameter, rule.Field);
    var value = Expression.Constant(rule.Data);
    var type = value.Type; 
    var containsmethod = type.GetMethod("Contains", new[] { typeof(string) });
    var call = Expression.Call(property, containsmethod, value);
    var lambda = Expression.Lambda<Func<T, bool>>(call, parameter);
    query = query.Where(lambda);
}
4

2 回答 2

2

您快到了,但是您的参数表达式应该是 type T,而不是String,您还缺少从Tname 之类的类型获取属性的表达式。

你应该大致拥有的是这个

val -> Expression.Constant(typeof(string), rule.Field)
parameter -> Expression.Parameter(typeof(T), "p")
property -> Expression.Property(parameter, "PropertyName")
contains -> Expression.Call(property, containsmethod, val)
equals true -> Expression.True or equals, something like that

我对所有这些都是徒手的,因此有效可能会有所不同。结果表达式应该是这样的

p => p.Name.Contains(val)
于 2013-01-30T22:09:59.793 回答
0

如果要创建Where查询,则必须先创建lambda然后调用Where查询并传递lambda。尝试这个:

Expression<Func<T, bool>> lambda = Expression.Lambda<Func<T, bool>>(call, parameter);
MethodCallExpression expression = Expression.Call(typeof(Queryable), "Where",
                                    new[] { typeof(T) }, query.Expression, lambda);
query = query.Provider.CreateQuery<T>(expression);

代替

var result = Expression.IsTrue(call);
query = query.Provider.CreateQuery<T>(result);
于 2013-01-31T10:53:16.667 回答