2009-04-08 19 views

Odpowiedz

62

Nie można po prostu rzucać między nie, ponieważ nie są one tym samym. Można jednak skutecznie dodać konwersję w drzewie wyrażenie:

using System; 
using System.Linq.Expressions; 

class Test 
{ 
    // This is the method you want, I think 
    static Expression<Func<TInput,object>> AddBox<TInput, TOutput> 
     (Expression<Func<TInput, TOutput>> expression) 
    { 
     // Add the boxing operation, but get a weakly typed expression 
     Expression converted = Expression.Convert 
      (expression.Body, typeof(object)); 
     // Use Expression.Lambda to get back to strong typing 
     return Expression.Lambda<Func<TInput,object>> 
      (converted, expression.Parameters); 
    } 

    // Just a simple demo 
    static void Main() 
    { 
     Expression<Func<string, DateTime>> x = text => DateTime.Now; 
     var y = AddBox(x);   
     object dt = y.Compile()("hi"); 
     Console.WriteLine(dt); 
    }   
} 
+0

@JonSkeet Expression.Convert nie zawsze jest dobrym pomysłem. Zobacz moją odpowiedź. – Rookian

8

na podstawie kodu z Jonem (dzięki btw) można wziąć krok dalej na pełną elastyczność:

public static Expression<Func<TModel, TToProperty>> Cast<TModel, TFromProperty, TToProperty>(Expression<Func<TModel, TFromProperty>> expression) 
{ 
    Expression converted = Expression.Convert(expression.Body, typeof(TToProperty)); 

    return Expression.Lambda<Func<TModel, TToProperty>>(converted, expression.Parameters); 
} 
+0

Czy jest jakiś przykład użycia tego? –

23

Odpowiedzi od Rob i Jon Skeet mają jeden problem.

dostać coś takiego x => Convert(x.PropertyName), ale często na przykład dla ASP.NET MVC chcesz wyrazem jak ten x => x.PropertyName

Więc Expression.Convert jest „zanieczyścić” ekspresja niektórych przypadkach.

Rozwiązanie:

public static class LambdaExpressionExtensions 
{ 
    public static Expression<Func<TInput, object>> ToUntypedPropertyExpression<TInput, TOutput> (this Expression<Func<TInput, TOutput>> expression) 
    { 
     var memberName = ((MemberExpression)expression.Body).Member.Name; 

     var param = Expression.Parameter(typeof(TInput)); 
     var field = Expression.Property(param, memberName); 
     return Expression.Lambda<Func<TInput, object>>(field, param); 
    } 
} 

Zastosowanie:

Expression<Func<T, DateTime>> expression = ...; 
Expression<Func<T, object>> expr = expression.ToUntypedPropertyExpression(); 
+7

To wydaje się nie działać. Na przykład, nie mogę przekonwertować 'Int32' na' object'; do tego służy wywołanie 'Convert'. Bez tego otrzymam "ArgumentException". Próbuję wersji 'DateTime', to samo. Jeśli to działało dla ciebie, domyślam się, że zrobiłeś to z typem referencyjnym. –

0

Wystarczy zdefiniować wyjście TResult jako przedmiot i skompilować wyrażenie, to działa dla wszystkich typów danych;

Expression<Func<string, object>> dateExp = text => DateTime.Now; 
object dt = dateExp.Compile()("hi"); 
Console.WriteLine(dt); 

Fiddle sample here

Powiązane problemy