2016-09-29 131 views
0

我试图动态转换这种类型的表达式:转换强类型的表达式匿名类型的返回

myDbSet.Select(x => new MyClass 
      { 
       IsSelected = x.ChildList.Any() 
      }) 

到:

myDbSet.Select(x => new 
      { 
       Item = x 
       IsSelected = x.ChildList.Any() 
      }) 

我会写,将采取强烈的扩展方法键入版本并转换为匿名,然后创建一个新表达式以执行如下操作:

myDbSet.Select(anonymousTransformedExpression).ToList(). 
     .Select(newGeneratedExpression) 

我想这newGeneratedExpression是:

myDbSet.Select(x => new 
      { 
       x.Item.IsSelected = x.IsSelected 
       return x.Item; 
      }) 

所以基本上等转换回回到一个强类型,但应用了“IsSelected”的价值。

问题是,我真的不能找到如何做到这一点的起点..

编辑

好吧,我意识到,问题不是那么清楚,我来一点点接近到迄今为止的一个主要问题的解决方案。下面是我的了:

public static IEnumerable<TModel> SelectWithUnmapped<TModel> (this IQueryable<TModel> source, Expression<Func<TModel, object>> assigner) where TModel : class, new() 
    { 
     var anonymousType = typeof(AnonymousType<TModel>); 
     var expression = Expression.New(anonymousType); 
     var parameter = Expression.Parameter(typeof(TModel), "x"); 


     //this part is hard coded to only take binding at position 0.. eventually will become dynamic 
     var originalBinding = ((MemberAssignment) ((MemberInitExpression) assigner.Body).Bindings[0]); 
     var originalExpression = originalBinding.Expression; 
     Expression conversion = Expression.Convert(originalExpression, typeof(object)); 

     var bindings = new[] 
     { 
      Expression.Bind(anonymousType.GetProperty("Item"), parameter), 
      //this is hardcoded test 
      Expression.Bind(anonymousType.GetProperty("Property1"), conversion) 
     }; 

     var body = Expression.MemberInit(expression, bindings); 
     var lambda = Expression.Lambda<Func<TModel, AnonymousType<TModel>>>(body, parameter); 




     var test = source.Select(lambda).ToList(); 

     return source; 
    } 

    class AnonymousType<TModel> 
    { 
     public TModel Item { get; set; } 
     public object Property1 { get; set; } 
     public object Property2 { get; set; } 
     public object Property3 { get; set; } 
     public object Property4 { get; set; } 
     public object Property5 { get; set; } 
     public object Property6 { get; set; } 
     public object Property7 { get; set; } 
     public object Property8 { get; set; } 
     public object Property9 { get; set; } 
     public object Property10 { get; set; } 
     public object Property11 { get; set; } 
     public object Property12 { get; set; } 
     public object Property13 { get; set; } 
     public object Property14 { get; set; } 
     public object Property15 { get; set; } 
     public object Property16 { get; set; } 
     public object Property17 { get; set; } 
     public object Property18 { get; set; } 
     public object Property19 { get; set; } 
     public object Property20 { get; set; } 
    } 
} 

当我尝试分配测试,我得到以下错误:参数“X”未在指定的LINQ绑定到实体查询表达式。

这是由于我的第二次绑定使用原始表达式的绑定表达式。但是两者都使用相同的参数名称“x”。我怎样才能确保我的新绑定真的知道x是新表达式的参数?

所以基本上到目前为止,我想参加,看起来像一个表达式:

x => new Test 
{ 
    PropertyTest = x.Blah.Count() 
} 

到:

x => new AnonymousType<Test>(){ 
    Item = x, 
    Property1 = x.Blah.Count() 
} 
+0

你需要创建一个ExpressionVisitor。见https://msdn.microsoft.com/en-us/library/bb882521(v=vs.90).aspx –

+0

但是你会遇到一个问题,因为匿名类型是在编译时定义的。它们实际上只是一个类型的隐式声明,所以在这里没有新的匿名类型,如果没有类型和构造函数,调用“Expression.New”将不起作用。 –

+0

我的问题不是很清楚,为了清晰我编辑。希望更清楚。我通过限制已知类型的对象属性来避免编译时出现的匿名类型问题。 –

回答

2

我终于完成了它。我想这只是耐心和试验和错误。帮助任何想要这样做的人。我不得不限制自己的财产数量..这可以很容易地添加。

下面的代码:

using System; 
using System.Collections.Generic; 
using System.Linq; 
using System.Linq.Expressions; 


public static class IQueryableExtensions 
{ 
    public static IEnumerable<TModel> SelectAndAssign<TModel>(this IQueryable<TModel> source, Expression<Func<TModel, object>> assigner) where TModel : class, new() 
    { 
     //get typed body of original expression 
     var originalBody = (MemberInitExpression)assigner.Body; 

     //list to store the new bindings we're creating for new expression 
     var newExpressionBindings = new List<MemberBinding>(); 
     var newExpressionReturnType = typeof(AnonymousType<TModel>); 

     //new param 
     var parameter = Expression.Parameter(typeof(TModel), "x"); 

     //base binding 
     newExpressionBindings.Add(Expression.Bind(newExpressionReturnType.GetProperty("Item"), parameter)); 

     //go through all the original expression's bindings 
     for (var i = 0; i < originalBody.Bindings.Count; i++) 
     { 
      var originalBinding = (MemberAssignment)originalBody.Bindings[i]; 
      var originalExpression = originalBinding.Expression; 

      var memberType = originalBinding.Expression.Type; 

      //create delegate based on the member type 
      var originalLambdaDelegate = typeof(Func<,>).MakeGenericType(typeof(TModel), memberType); 

      //create lambda from that delegate 
      var originalLambda = Expression.Lambda(originalLambdaDelegate, originalExpression, assigner.Parameters[0]); 

      //create a AnonymousVar<MemberType> from the type of the member (to please EF unable to assign bool to object directly) 
      //start with getting the generic type 
      var genericMemberType = typeof(AnonymousVar<>).MakeGenericType(memberType); 
      //then create teh delegate 
      var genericMemberTypeDelegate = typeof(Func<>).MakeGenericType(genericMemberType); 
      //Now create an expression with a binding for that object to assign its Property (strongly typed now from the generic declaration) 
      var genericInstantiationExpression = Expression.New(genericMemberType); 
      //the binding.. using the original expression expression 
      var genericInstantiationBinding = Expression.Bind(genericMemberType.GetProperty("Property"), originalLambda.Body); 
      // create the body 
      var genericInstantiationBody = Expression.MemberInit(genericInstantiationExpression, genericInstantiationBinding); 

      //now we need to recreate a lambda for this 
      var newBindingExpression = Expression.Lambda(genericMemberTypeDelegate, genericInstantiationBody); 

      //Create the binding and add it to the new expression bindings 
      newExpressionBindings.Add(Expression.Bind(newExpressionReturnType.GetProperty("Property" + (i + 1)), newBindingExpression.Body)); 
     } 

     //start creating the new expression 
     var expression = Expression.New(newExpressionReturnType); 

     //create new expression body with bindings 
     var body = Expression.MemberInit(expression, newExpressionBindings); 

     //The actual new expression lambda 
     var newLambda = Expression.Lambda<Func<TModel, AnonymousType<TModel>>>(body, parameter); 

     // replace old lambda param with new one 
     var replacer = new ParameterReplacer(assigner.Parameters[0], newLambda.Parameters[0]); // replace old lambda param with new 

     //new lambda with fixed params 
     newLambda = Expression.Lambda<Func<TModel, AnonymousType<TModel>>>(replacer.Visit(newLambda.Body), newLambda.Parameters[0]); 

     //now that we have all we need form the server, we materialize the list 
     var materialized = source.Select(newLambda).ToList(); 

     //typed return parameter 
     var typedReturnParameter = Expression.Parameter(typeof(AnonymousType<TModel>), "x"); 

     //Lets assign all those custom properties back into the original object type 
     var expressionLines = new List<Expression>(); 
     for (var i = 0; i < originalBody.Bindings.Count; i++) 
     { 
      var originalBinding = (MemberAssignment)originalBody.Bindings[i]; 
      var itemPropertyExpression = Expression.Property(typedReturnParameter, "Item"); 
      var bindingPropertyExpression = Expression.Property(itemPropertyExpression, originalBinding.Member.Name); 

      var memberType = originalBinding.Expression.Type; 
      var valuePropertyExpression = Expression.Convert(Expression.Property(typedReturnParameter, "Property" + (i + 1)), typeof(AnonymousVar<>).MakeGenericType(memberType)); 
      var memberValuePropertyExpression = Expression.Property(valuePropertyExpression, "Property"); 

      var equalExpression = Expression.Assign(bindingPropertyExpression, memberValuePropertyExpression); 
      expressionLines.Add(equalExpression); 
     } 
     var returnTarget = Expression.Label(typeof(TModel)); 
     expressionLines.Add(Expression.Return(returnTarget, Expression.Property(typedReturnParameter, "Item"))); 
     expressionLines.Add(Expression.Label(returnTarget, Expression.Constant(null, typeof(TModel)))); 
     var finalExpression = Expression.Block(expressionLines); 

     var typedReturnLambda = Expression.Lambda<Func<AnonymousType<TModel>, TModel>>(finalExpression, typedReturnParameter).Compile(); 

     return materialized.Select(typedReturnLambda); 
    } 

    class AnonymousVar<TModel> 
    { 
     public TModel Property { get; set; } 
    } 

    class AnonymousType<TModel> 
    { 
     public TModel Item { get; set; } 

     public object Property1 { get; set; } 
     public object Property2 { get; set; } 
     public object Property3 { get; set; } 
     public object Property4 { get; set; } 
     public object Property5 { get; set; } 
     public object Property6 { get; set; } 
     public object Property7 { get; set; } 
     public object Property8 { get; set; } 
     public object Property9 { get; set; } 
     public object Property10 { get; set; } 
     public object Property11 { get; set; } 
     public object Property12 { get; set; } 
     public object Property13 { get; set; } 
     public object Property14 { get; set; } 
     public object Property15 { get; set; } 
     public object Property16 { get; set; } 
     public object Property17 { get; set; } 
     public object Property18 { get; set; } 
     public object Property19 { get; set; } 
     public object Property20 { get; set; } 
    } 


    class ParameterReplacer : ExpressionVisitor 
    { 
     private ParameterExpression from; 
     private ParameterExpression to; 

     public ParameterReplacer(ParameterExpression from, ParameterExpression to) 
     { 
      this.from = from; 
      this.to = to; 
     } 
     protected override Expression VisitParameter(ParameterExpression node) 
     { 
      return base.VisitParameter(node == this.from ? this.to : node); 
     } 
    } 
} 
相关问题