我注意到在C#中处理lambda函数和匿名委托时,一些有用但不起作用的例子。这是怎么回事?
class Test : Control {
void testInvoke() {
// The best overloaded method match for 'Invoke' has some invalid arguments
Invoke(doSomething);
// Cannot convert anonymous method to type 'System.Delegate' because it is not a delegate type
Invoke(delegate { doSomething(); });
// OK
Invoke((Action)doSomething);
// OK
Invoke((Action)delegate { doSomething(); });
// Cannot convert lambda expression to type 'System.Delegate' because it is not a delegate type
Invoke(() => doSomething());
// OK
Invoke((Action)(() => doSomething()));
}
void testQueueUserWorkItem() {
// The best overloaded method match for 'QueueUserWorkItem' has some invalid arguments
ThreadPool.QueueUserWorkItem(doSomething);
// OK
ThreadPool.QueueUserWorkItem(delegate { doSomething(); });
// The best overloaded method match for 'QueueUserWorkItem' has some invalid arguments
ThreadPool.QueueUserWorkItem((Action)doSomething);
// No overload for 'doSomething' matches delegate 'WaitCallback'
ThreadPool.QueueUserWorkItem((WaitCallback)doSomething);
// OK
ThreadPool.QueueUserWorkItem((WaitCallback)delegate { doSomething(); });
// Delegate 'WaitCallback' does not take '0' arguments
ThreadPool.QueueUserWorkItem(() => doSomething());
// OK
ThreadPool.QueueUserWorkItem(state => doSomething());
}
void doSomething() {
// ...
}
}
那是很多例子。我想我的问题如下:
为什么Invoke
总是拒绝lambda函数或匿名委托,但ThreadPool.QueueUserWorkItem
还不错?
究竟是什么“无法将匿名方法转换为'System.Delegate',因为它不是委托类型”是指“无论如何?”
为什么ThreadPool.QueueUserWorkItem
接受没有参数的匿名委托,而不接受没有参数的lambda表达式?
答案 0 :(得分:9)
ThreadPool.QueueUserWorkItem
在其签名中有特定的委托;调用只有Delegate
。 Lambda表达式和匿名方法只能转换为特定的委托类型。
这只是一个错误的错误消息。这意味着,“我不确切地知道您要转换为哪种委托类型。”
您使用的是匿名方法,而且没有参数列表,可以将其转换为不使用out / ref参数的任何委托类型。如果您尝试delegate() { ... }
(即显式空参数列表),那么它将无效。匿名方法的“我不关心参数”能力是他们拥有的唯一特性,而lambda表达式则没有。
最容易在简单分配的背景下展示所有这些,IMO:
// Doesn't work: no specific type
Delegate d = () => Console.WriteLine("Bang");
// Fine: we know the exact type to convert to
Action a = () => Console.WriteLine("Yay");
// Doesn't work: EventHandler isn't parameterless; we've specified 0 parameters
EventHandler e1 = () => Console.WriteLine("Bang");
EventHandler e2 = delegate() { Console.WriteLine("Bang again"); };
// Works: we don't care about parameter lists
EventHandler e = delegate { Console.WriteLine("Lambdas can't do this"); };