2011-09-09 70 views
21

我想模仿左外連接here,但使用動態linq擴展方法。我有:如何使用Dynamic Linq進行左外連接?

public static IQueryable SelectMany(this IQueryable source, string selector, 
    string resultsSelector, params object[] values) 
{ 
    if (source == null) throw new ArgumentNullException("source"); 
    if (selector == null) throw new ArgumentNullException("selector"); 

    // Parse the lambda 
    LambdaExpression lambda = DynamicExpression.ParseLambda(
     source.ElementType, null, selector, values); 

    // Fix lambda by recreating to be of correct Func<> type in case 
    // the expression parsed to something other than IEnumerable<T>. 
    // For instance, a expression evaluating to List<T> would result 
    // in a lambda of type Func<T, List<T>> when we need one of type 
    // an Func<T, IEnumerable<T> in order to call SelectMany(). 
    Type inputType = source.Expression.Type.GetGenericArguments()[0]; 
    Type resultType = lambda.Body.Type.GetGenericArguments()[0]; 
    Type enumerableType = typeof(IEnumerable<>).MakeGenericType(resultType); 
    Type delegateType = typeof(Func<,>).MakeGenericType(inputType, 
     enumerableType); 
    lambda = Expression.Lambda(delegateType, lambda.Body, lambda.Parameters); 

    ParameterExpression[] parameters = new ParameterExpression[] { 
     Expression.Parameter(source.ElementType, "outer"), 
     Expression.Parameter(resultType, "inner") 
    }; 

    LambdaExpression resultsSelectorLambda = DynamicExpression.ParseLambda(
     parameters, null, resultsSelector, values); 

    // Create the new query 
    return source.Provider.CreateQuery(Expression.Call(typeof(Queryable), 
     "SelectMany", new Type[] { 
      source.ElementType, 
      resultType, 
      resultsSelectorLambda.Body.Type 
     }, source.Expression, Expression.Quote(lambda), 
     Expression.Quote(resultsSelectorLambda)));    
} 

和:

public static IQueryable GroupJoin(this IQueryable outer, IEnumerable inner, 
    string outerKeySelector, string innerKeySelector, string resultSelector, 
    params object[] values) 
{ 
    Type innerElementType = inner.AsQueryable().ElementType; 

    var outerParameter = Expression.Parameter(outer.ElementType, "outer"); 
    var innerParameter = Expression.Parameter(innerElementType, "inner"); 
    var groupParameter = Expression.Parameter(typeof(IEnumerable<>) 
     .MakeGenericType(innerElementType), "group"); 

    var outerLambda = DynamicExpression.ParseLambda(new[] { outerParameter }, 
     null, outerKeySelector, values); 
    var innerLambda = DynamicExpression.ParseLambda(new[] { innerParameter }, 
     outerLambda.Body.Type, innerKeySelector, values); 
    var resultLambda = DynamicExpression.ParseLambda(new[] { 
     outerParameter, groupParameter }, null, resultSelector, values); 

    return outer.Provider.CreateQuery(Expression.Call(typeof(Queryable), 
     "GroupJoin", new[] { outer.ElementType, innerElementType, 
     outerLambda.Body.Type, resultLambda.Body.Type }, 
     outer.Expression, Expression.Constant(inner), 
     Expression.Quote(outerLambda), Expression.Quote(innerLambda), 
     Expression.Quote(resultLambda))); 
} 

然而,當我倒下是內SelectMany

+1

你指的是哪這裏的DynamicExpression實施打個電話? – arviman

回答

9

DefaultIfEmpty添加void DefaultIfEmpty();interface IEnumerableSignatures

然後使用

public static object DefaultIfEmpty(this IQueryable source) 
{ 
    if (source == null) throw new ArgumentNullException("source"); 
     return source.Provider.Execute(
    Expression.Call(
     typeof(Queryable), "DefaultIfEmpty", 
     new Type[] { source.ElementType }, 
     source.Expression)); 
} 

然後你有一個像

var qry = Foo.GroupJoin(Bar, "outer.Id", "inner.Id", "new(outer.Id as Foo, group as Bars)").SelectMany("Bars.DefaultIfEmpty()", "new(outer.Foo as Foo, inner as Bar)"); 
+0

你可以給你簡要的信息嗎?我需要使用這個:( – Licentia