如何使用条件三元运算符在lambdas之间有条件地分配Func <>?

BFr*_*ree 47 c# lambda conditional-operator

通常,在使用条件运算符时,这是语法:

int x = 6;
int y = x == 6 ? 5 : 9;
Run Code Online (Sandbox Code Playgroud)

没有什么花哨的,非常直接的.

现在,让我们在将Lambda分配给Func类型时尝试使用它.让我解释:

Func<Order, bool> predicate = id == null
    ? p => p.EmployeeID == null
    : p => p.EmployeeID == id;
Run Code Online (Sandbox Code Playgroud)

这是相同的语法,应该工作吗?对?出于某种原因,没有.编译器提供了这个不错的神秘消息:

错误1无法确定条件表达式的类型,因为"lambda表达式"和"lambda表达式"之间没有隐式转换

然后我继续改变语法,这样它确实有效:

Func<Order, bool> predicate = id == null
    ? predicate = p => p.EmployeeID == null
    : predicate = p => p.EmployeeID == id;
Run Code Online (Sandbox Code Playgroud)

我只是好奇为什么它不能以第一种方式工作?

(旁注:我最终不需要这段代码,因为我发现在将int值与null进行比较时,只需使用object.Equals)

Jon*_*eet 45

您可以将lambda表达式转换为特定的目标委托类型,但是为了确定条件表达式的类型,编译器需要知道每个第二个和第三个操作数的类型.虽然它们都只是"lambda表达式",但是没有从一个转换到另一个,所以编译器不能做任何有用的事情.

我建议不要使用作业 - 演员阵容更明显:

Func<Order, bool> predicate = id == null 
    ? (Func<Order, bool>) (p => p.EmployeeID == null)
    : p => p.EmployeeID == id;
Run Code Online (Sandbox Code Playgroud)

请注意,您只需要为一个操作数提供它,因此编译器可以从另一个lambda表达式执行转换.


Jak*_*ake 6

C#编译器无法推断创建的lambda表达式的类型,因为它首先处理三元组然后处理赋值.你也可以这样做:

Func<Order, bool> predicate = 
    id == null ? 
        new Func<Order,bool>(p => p.EmployeeID == null) :
        new Func<Order,bool>(p => p.EmployeeID == id);
Run Code Online (Sandbox Code Playgroud)

但这很糟糕,你也可以试试

Func<Order, bool> predicate = 
    id == null ? 
        (Order p) => p.EmployeeID == null :
        (Order p) => p.EmployeeID == id;
Run Code Online (Sandbox Code Playgroud)

  • 后者不起作用,因为编译器不知道是否要转换为委托或表达式树(或者说,例如,Func <Order,object>也可以). (4认同)