2013-03-08 10 views
7

Sto provando a creare un'espressione filtro per filtrare i dati dal database.Albero delle espressioni e condizione AND

ho scritto la seguente estensione per costruire l'espressione dinamicamente a seconda dei parametri di filtro selezionati:

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; 
} 

lo sto usando in questo modo:

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

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

È perfettamente compatibile con NHibernate , ma quando sto usando questo codice con Entity Framework 5 non riesce con il seguente messaggio di eccezione:

Il tipo di nodo di espressione LINQ "Invoke" non è supportato in LINQ alle entità.

È una soluzione che recupera l'intera collezione dal database e poi applica condizioni di filtro attraverso il IEnumerable.Where(Func<T1, T2> filterClause), ma non hanno bisogno di tutti i dati solo per ottenere un record, mentre Expression<Func<T1, T2>> espressioni sono tradotti direttamente in SQL.

Esiste un modo semplice per far funzionare questo codice con Entity Framework?

risposta

6

Prova questa implementazione da Domain Oriented N-Layered .NET 4.0 Sample App (c'è anche implementazione di 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

Si prega di trovare la soluzione nel post del blog di seguito:

A universal PredicateBuilder

L'autore di questo post del blog fornisce un altro per l'attuazione del metodo di "Add" per unire 2 espressione.

/// <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 solo" risposte tendono ad essere visto di buon occhio qui. Potresti almeno fornire un riassunto o altra indicazione di * perché * sarebbe utile. C'è anche la possibilità che qualunque cosa tu stia collegando potrebbe non essere più disponibile quando le persone leggono più tardi le tue risposte. –