2013-03-08 9 views
7

Próbuję utworzyć wyrażenie filtru w celu filtrowania danych z bazy danych.Drzewo wyrażeń i warunek AND

ja napisałem następujące rozszerzenia do budowania wyraz dynamicznie w zależności od wybranych parametrów filtra:

public static Expression<Func<T, bool>> And<T>(
     this Expression<Func<T, bool>> leftExpression, 
     Expression<Func<T, bool>> rightExpression) 
{ 
    var invocationExpression = Expression.Invoke(rightExpression, leftExpression.Parameters.Cast<Expression>()); 
    var andExpression = Expression.Lambda<Func<T, bool>>(
     Expression.AndAlso(leftExpression.Body, invocationExpression), 
     leftExpression.Parameters); 

    return andExpression; 
} 

używam go w taki sposób:

Expression<Func<MyObject, bool>> expression = x => true; 

if(MyFilter.SomeParam) { 
    expression = expression.And(x=>x.MyProperty == MyFilter.SomeParam); 
} 

działa świetnie z NHibernate , ale gdy używam tego kodu w Entity Framework 5, kończy się niepowodzeniem z następującym komunikatem wyjątku:

Typ węzła wyrażeń LINQ "Wywołanie" nie jest obsługiwany w LINQ do jednostek.

Jest jedno obejście, które pobiera całą kolekcję z bazy danych, a następnie stosuje się warunki filtrowania przez IEnumerable.Where(Func<T1, T2> filterClause), ale nie potrzebują wszystkie dane, aby dostać jeden rekord podczas Expression<Func<T1, T2>> wyrażenia są tłumaczone bezpośrednio do SQL.

Czy istnieje prosty sposób, aby ten kod działał z Entity Framework?

Odpowiedz

6

Spróbuj realizację od Domain Oriented N-Layered .NET 4.0 Sample App (jest również realizacja specification pattern):

public static class ExpressionBuilder 
{ 
    public static Expression<T> Compose<T>(this Expression<T> first, Expression<T> second, Func<Expression, Expression, Expression> merge) 
    { 
     var map = first.Parameters.Select((f, i) => new { f, s = second.Parameters[i] }).ToDictionary(p => p.s, p => p.f); 
     var secondBody = ParameterRebinder.ReplaceParameters(map, second.Body); 
     return Expression.Lambda<T>(merge(first.Body, secondBody), first.Parameters); 
    } 

    public static Expression<Func<T, bool>> And<T>(this Expression<Func<T, bool>> first, Expression<Func<T, bool>> second) 
    { 
     return first.Compose(second, Expression.And); 
    } 

    public static Expression<Func<T, bool>> Or<T>(this Expression<Func<T, bool>> first, Expression<Func<T, bool>> second) 
    { 
     return first.Compose(second, Expression.Or); 
    } 

} 

public class ParameterRebinder : ExpressionVisitor 
{ 
    private readonly Dictionary<ParameterExpression, ParameterExpression> map; 

    public ParameterRebinder(Dictionary<ParameterExpression, ParameterExpression> map) 
    { 
     this.map = map ?? new Dictionary<ParameterExpression, ParameterExpression>(); 
    } 

    public static Expression ReplaceParameters(Dictionary<ParameterExpression, ParameterExpression> map, Expression exp) 
    { 
     return new ParameterRebinder(map).Visit(exp); 
    } 

    protected override Expression VisitParameter(ParameterExpression p) 
    { 
     ParameterExpression replacement; 
     if (map.TryGetValue(p, out replacement)) 
     { 
      p = replacement; 
     } 

     return base.VisitParameter(p); 
    } 

} 
2

Należy znaleźć rozwiązanie w poniższej blogu:

A universal PredicateBuilder

Autor tego blogu zapewnić inny z realizacją „Dodaj” metody łączenia 2 wyraz.

/// <summary> 
/// Enables the efficient, dynamic composition of query predicates. 
/// </summary> 
public static class PredicateBuilder 
{ 
    /// <summary> 
    /// Creates a predicate that evaluates to true. 
    /// </summary> 
    public static Expression<Func<T, bool>> True<T>() { return param => true; } 

    /// <summary> 
    /// Creates a predicate that evaluates to false. 
    /// </summary> 
    public static Expression<Func<T, bool>> False<T>() { return param => false; } 

    /// <summary> 
    /// Creates a predicate expression from the specified lambda expression. 
    /// </summary> 
    public static Expression<Func<T, bool>> Create<T>(Expression<Func<T, bool>> predicate) { return predicate; } 

    /// <summary> 
    /// Combines the first predicate with the second using the logical "and". 
    /// </summary> 
    public static Expression<Func<T, bool>> And<T>(this Expression<Func<T, bool>> first, Expression<Func<T, bool>> second) 
    { 
     return first.Compose(second, Expression.AndAlso); 
    } 

    /// <summary> 
    /// Combines the first predicate with the second using the logical "or". 
    /// </summary> 
    public static Expression<Func<T, bool>> Or<T>(this Expression<Func<T, bool>> first, Expression<Func<T, bool>> second) 
    { 
     return first.Compose(second, Expression.OrElse); 
    } 

    /// <summary> 
    /// Negates the predicate. 
    /// </summary> 
    public static Expression<Func<T, bool>> Not<T>(this Expression<Func<T, bool>> expression) 
    { 
     var negated = Expression.Not(expression.Body); 
     return Expression.Lambda<Func<T, bool>>(negated, expression.Parameters); 
    } 

    /// <summary> 
    /// Combines the first expression with the second using the specified merge function. 
    /// </summary> 
    static Expression<T> Compose<T>(this Expression<T> first, Expression<T> second, Func<Expression, Expression, Expression> merge) 
    { 
     // zip parameters (map from parameters of second to parameters of first) 
     var map = first.Parameters 
      .Select((f, i) => new { f, s = second.Parameters[i] }) 
      .ToDictionary(p => p.s, p => p.f); 

     // replace parameters in the second lambda expression with the parameters in the first 
     var secondBody = ParameterRebinder.ReplaceParameters(map, second.Body); 

     // create a merged lambda expression with parameters from the first expression 
     return Expression.Lambda<T>(merge(first.Body, secondBody), first.Parameters); 
    } 

    class ParameterRebinder : ExpressionVisitor 
    { 
     readonly Dictionary<ParameterExpression, ParameterExpression> map; 

     ParameterRebinder(Dictionary<ParameterExpression, ParameterExpression> map) 
     { 
      this.map = map ?? new Dictionary<ParameterExpression, ParameterExpression>(); 
     } 

     public static Expression ReplaceParameters(Dictionary<ParameterExpression, ParameterExpression> map, Expression exp) 
     { 
      return new ParameterRebinder(map).Visit(exp); 
     } 

     protected override Expression VisitParameter(ParameterExpression p) 
     { 
      ParameterExpression replacement; 

      if (map.TryGetValue(p, out replacement)) 
      { 
       p = replacement; 
      } 

      return base.VisitParameter(p); 
     } 
    } 
} 
+1

"Link tylko" odpowiedzi wydają się być mile widziana tutaj. Czy mógłbyś przynajmniej podać streszczenie lub inne wskazanie * dlaczego * byłoby użyteczne. Istnieje również możliwość, że cokolwiek łączysz, nie będzie już dostępne, gdy ludzie później przeczytają twoje odpowiedzi. –

Powiązane problemy