无法从用法从表达式返回表达式的用途来推断方法的类型实参

时间:2019-06-20 17:29:51

标签: c# linq

我正在尝试从Select返回一个表达式,所以我有一个表达式返回一个表达式(我想-也许是lambda返回一个表达式,我不确定这个术语)

如果我创建了一个明确为表达式指定类型的变量,则它可以正常工作

housingDivisions.Select(id => {
    Expression<Func<Document, bool>> expression = d => d.HousingDivisions.Any(h => h.HousingDivisionId == id);
    return expression;
})

然而,这段代码似乎是等效的(而且不那么冗长)

housingDivisions.Select(id => d => d.HousingDivisions.Any(h => h.HousingDivisionId == id))

但会导致编译错误:

  

错误CS0411无法从用法中推断出方法'Enumerable.Select(IEnumerable,Func)'的类型参数。尝试显式指定类型参数。

有什么方法可以编译它而不必创建不必要的变量?

1 个答案:

答案 0 :(得分:1)

问题在于类型d和委托类型d => ...都是未知的,无法在任何地方进行假设。您可以通过实例化委托类型来一次性解决这两个问题:

housingDivisions.Select(id => new Func<Document, bool>(d => d.HousingDivisions.Any(h => h.HousingDivisionId == id)))

如果您绝对需要将类型设为表达式树,则可以通过强制转换为它:

housingDivisions.Select(id => (Expression<Func<Document, bool>>)(d => d.HousingDivisions.Any(h => h.HousingDivisionId == id)))

编译器建议显式指定参数。我个人认为这种方式比较罗word。但它看起来像这样:

housingDivisions.Select<int, Expression<Func<Document, bool>>>(id => d => d.HousingDivisions.Any(h => h.HousingDivisionId == id));

以下是用于选择表达式的扩展方法:

public static IEnumerable<Expression<Func<ExprArg, Result>>> SelectExpr<TSource, ExprArg, Result>(this IEnumerable<TSource> source, Func<TSource, ExprArg, Result> func)
{
    return source.Select((o) => (Expression<Func<ExprArg, Result>>)((arg) => func(o, arg)));
}


// Use:
housingDivisions.SelectExpr((int id, Document d) => d.HousingDivisions.Any(h => h.HousingDivisionId == id));