我正在尝试从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)'的类型参数。尝试显式指定类型参数。
有什么方法可以编译它而不必创建不必要的变量?
答案 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));