3

一段时间以来,我一直在摆弄动态 LINQ,但我还没有了解它的秘密。

我有一个要解析的表达式,如下所示:

"document.LineItems.Select(i => i.Credit).Sum();"

在解析这个的过程中,我需要在 LineItems Collection 上调用 Select 函数。我正在使用 Expression.Call 的工厂方法:

Expression.Call(
    typeof(Queryable),
    "Select",
    new Type[] { typeof(LineItem), typeof(decimal?) },
    expr,
    Expression.Lambda(expression, new ParameterExpression[] { Expression.Parameter(typeof(LineItem) }));

此时此刻

expr:          document.LineItems    [ICollection<LineItem>]  
expression:    LineItem.Credit       [decimal?]

这些都还没有实现。我现在正在构建表达式树。

现在,问题:

此 Expresssion.Call 抛出异常:“类型 'System.Linq.Queryable' 上没有通用方法 'Select' 与提供的类型参数和参数兼容”;

我通过更改 Expression.Call 的第一个参数在“System.Linq.Enumerable”而不是“Queryable”中查找“Select”来轻松解决它。

但是,这不是我想要的。我不希望所有的 LineItems 都只用于计算 Sum(),这显然是 Enumerable 的情况。我希望 Queryable 工作。

此外,对于解析的最后一部分 - Sum(),我还需要使用 Enumerable Sum(),因为 Queryable Sum() 会抛出相同的异常。

我检查了 MSDN,'Select' 函数的两个签名是相同的,所以我真的不明白为什么一个工作而另一个不工作。

任何帮助或指示将不胜感激。

问候,

4

1 回答 1

4

问题是它LineItems是一个ICollection<LineItem>,而第一个参数Queryable.Select需要一个IQueryable<LineItem>ICollection<T>只有实现IEnumerable<T>,这就是为什么它适用于Enumerable.Select.

您需要将类型更改exprIQueryable<LineItem>

您应该能够使用该Queryable.AsQueryable方法执行此操作。以下函数创建一个表达式来对给定Document实例的信用属性值求和:

public static Expression<Func<decimal?>> CreateSumLineItemsExpr(Document document)
{
    var docExpr = Expression.Constant(document);
    var itemsExpr = Expression.Property(docExpr, "LineItems");

    Expression<Func<LineItem, decimal?>> selector = i => i.Credit;
    var queryableExpr = Expression.Call(typeof(Queryable), "AsQueryable", new[] { typeof(LineItem) }, itemsExpr);
    var selectExpr = Expression.Call(typeof(Queryable), "Select", new[] { typeof(LineItem), typeof(decimal?) }, queryableExpr, selector);
    var sumExpr = Expression.Call(typeof(Queryable), "Sum", null, selectExpr);

    return Expression.Lambda<Func<decimal?>>(sumExpr);
}
于 2012-12-08T13:31:25.637 回答