2010-02-19 7 views
9

Ok, la mia ipotesi è che questa risposta sia già stata già da qualche parte, e io non sono abbastanza familiare con la sintassi che ancora non capisco, quindi abbi pazienza con me.Esegui dinamicamente l'espressione Lambq di Lambda

Gli utenti della mia app Web devono filtrare un lungo elenco di elementi in un gridview, a cui si accede tramite un linqdatasource. Sto usando l'evento OnSelecting per filtrare ulteriormente gli elementi. Voglio filtrare quegli elementi in base alle selezioni effettuate dagli utenti in DropDownLists.

Ad esempio, selezionare "Title" "Contiene" "Fred" Ciò si traduce in

e.Result = dbContext.Opps.Where(opp => opp.Title.Contains("Fred")); 

Or "Descrizione" "Non contiene" "Alpha" risultati in

e.Result = dbContext.Opps.Where(opp => !opp.Description.Contains("Alpha")); 

Mi piacerebbe creare quell'espressione (System.Linq.Expressions.Expression>) dinamicamente, invece di avere espressioni di commutazione annidate per generarlo, poiché ci sono un certo numero di campi che voglio controllare, e voglio anche usare il Inizia con e finisce con il controllo S. Se potessi costruire l'espressione come una stringa, in questo modo:

string stringExpression = string.Format("opp => opp.{0}.{1}(\"{2}\")", 
    ddlCustomFilter.SelectedValue, 
    ddlFilterType.SelectedValue, 
    txtFilterText.Text); 

E poi in qualche modo lo hanno essere convertito in un'espressione ... è possibile? O dovrei semplicemente mordere il proiettile e generare tutte le istruzioni switch() necessarie per creare le varie espressioni?

+0

http://stackoverflow.com/questions/1810808/linq-the-dynamic-query-or -how-to-get-a-unknown-number-of-columns – Luiscencio

risposta

8

Si potrebbe sicuramente creare l'espressione in modo dinamico, ma prima prenderei in considerazione l'utilizzo di Dynamic LINQ, anche se potrebbe non essere possibile utilizzare Contiene. Inoltre, è possibile prendere in considerazione l'utilizzo di PredicateBuilder per aggiungere query complesse in modo additivo.

+0

Accetto di utilizzare la libreria Dynamic Linq. Un grande brutto funzionerà oggi, ma più difficile da mantenere/gestire lungo la strada. –

+0

Linq dinamico funziona perfettamente. Gestisce anche Field.Contains e! Field.Contains. Il che è buono, perché non riesco a creare teste o code di PredicateBuilder. – Dave

+0

Devo scaricare qualcosa per ottenere PredicateBuilder su vs210 .net4.0? – guiomie

4

provare questo codice ...

chiamata ToExpression Method() ....

public static Expression<Func<T, bool>> ToExpression<T>(string andOrOperator, string propName, string opr, string value, Expression<Func<T, bool>> expr = null) 
    { 
     Expression<Func<T, bool>> func = null; 
     try 
     { 
      ParameterExpression paramExpr = Expression.Parameter(typeof(T)); 
      var arrProp = propName.Split('.').ToList(); 
      Expression binExpr = null; 
      string partName = string.Empty; 
      arrProp.ForEach(x => 
      { 
       Expression tempExpr = null; 
       partName = partName.IsNull() ? x : partName + "." + x; 
       if (partName == propName) 
       { 
        var member = NestedExprProp(paramExpr, partName); 
        var type = member.Type.Name == "Nullable`1" ? Nullable.GetUnderlyingType(member.Type) : member.Type; 
        tempExpr = ApplyFilter(opr, member, Expression.Convert(ToExprConstant(type, value), member.Type)); 
       } 
       else 
        tempExpr = ApplyFilter("!=", NestedExprProp(paramExpr, partName), Expression.Constant(null)); 
       if (binExpr != null) 
        binExpr = Expression.AndAlso(binExpr, tempExpr); 
       else 
        binExpr = tempExpr; 
      }); 
      Expression<Func<T, bool>> innerExpr = Expression.Lambda<Func<T, bool>>(binExpr, paramExpr); 
      if (expr != null) 
       innerExpr = (andOrOperator.IsNull() || andOrOperator == "And" || andOrOperator == "AND" || andOrOperator == "&&") ? innerExpr.And(expr) : innerExpr.Or(expr); 
      func = innerExpr; 
     } 
     catch { } 
     return func; 
    } 

    private static MemberExpression NestedExprProp(Expression expr, string propName) 
    { 
     string[] arrProp = propName.Split('.'); 
     int arrPropCount = arrProp.Length; 
     return (arrPropCount > 1) ? Expression.Property(NestedExprProp(expr, arrProp.Take(arrPropCount - 1).Aggregate((a, i) => a + "." + i)), arrProp[arrPropCount - 1]) : Expression.Property(expr, propName); 
    } 

    private static Expression ToExprConstant(Type prop, string value) 
    { 
     if (value.IsNull()) 
      return Expression.Constant(value); 
     object val = null; 
     switch (prop.FullName) 
     { 
      case "System.Guid": 
       val = value.ToGuid(); 
       break; 
      default: 
       val = Convert.ChangeType(value, Type.GetType(prop.FullName)); 
       break; 
     } 
     return Expression.Constant(val); 
    } 

    private static Expression ApplyFilter(string opr, Expression left, Expression right) 
    { 
     Expression InnerLambda = null; 
     switch (opr) 
     { 
      case "==": 
      case "=": 
       InnerLambda = Expression.Equal(left, right); 
       break; 
      case "<": 
       InnerLambda = Expression.LessThan(left, right); 
       break; 
      case ">": 
       InnerLambda = Expression.GreaterThan(left, right); 
       break; 
      case ">=": 
       InnerLambda = Expression.GreaterThanOrEqual(left, right); 
       break; 
      case "<=": 
       InnerLambda = Expression.LessThanOrEqual(left, right); 
       break; 
      case "!=": 
       InnerLambda = Expression.NotEqual(left, right); 
       break; 
      case "&&": 
       InnerLambda = Expression.And(left, right); 
       break; 
      case "||": 
       InnerLambda = Expression.Or(left, right); 
       break; 
      case "LIKE": 
       InnerLambda = Expression.Call(left, typeof(string).GetMethod("Contains", new Type[] { typeof(string) }), right); 
       break; 
      case "NOTLIKE": 
       InnerLambda = Expression.Not(Expression.Call(left, typeof(string).GetMethod("Contains", new Type[] { typeof(string) }), right)); 
       break; 
     } 
     return InnerLambda; 
    } 

    public static Expression<Func<T, object>> PropExpr<T>(string PropName) 
    { 
     ParameterExpression paramExpr = Expression.Parameter(typeof(T)); 
     var tempExpr = Extentions.NestedExprProp(paramExpr, PropName); 
     return Expression.Lambda<Func<T, object>>(Expression.Convert(Expression.Lambda(tempExpr, paramExpr).Body, typeof(object)), paramExpr); 

    } 
    public static IQueryOver<T, T> OrderBy<T>(this IQueryOver<T, T> Collection, string sidx, string sord) 
    { 
     return sord == "asc" ? Collection.OrderBy(NHibernate.Criterion.Projections.Property(sidx)).Asc : Collection.OrderBy(NHibernate.Criterion.Projections.Property(sidx)).Desc; 
    } 

    public static Expression<Func<T, TResult>> And<T, TResult>(this Expression<Func<T, TResult>> expr1, Expression<Func<T, TResult>> expr2) 
    { 
     var invokedExpr = Expression.Invoke(expr2, expr1.Parameters.Cast<Expression>()); 
     return Expression.Lambda<Func<T, TResult>>(Expression.AndAlso(expr1.Body, invokedExpr), expr1.Parameters); 
    } 

    public static Expression<Func<T, bool>> Or<T>(this Expression<Func<T, bool>> expr1, Expression<Func<T, bool>> expr2) 
    { 
     var invokedExpr = Expression.Invoke(expr2, expr1.Parameters.Cast<Expression>()); 
     return Expression.Lambda<Func<T, bool>>(Expression.OrElse(expr1.Body, invokedExpr), expr1.Parameters); 
    } 
Problemi correlati