I’ve noticed some examples of things that work and don’t work when dealing with lambda functions and anonymous delegates in C#. What’s going on here?
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() {
// ...
}
}
Well that’s a lot of examples. I guess my questions are the following:
-
Why does
Invokealways refuse a lambda function or an anonymous delegate, yetThreadPool.QueueUserWorkItemdoes just fine? -
What the heck does “Cannot convert anonymous method to type ‘System.Delegate’ because it is not a delegate type” mean anyway?
-
Why does
ThreadPool.QueueUserWorkItemaccept an anonymous delegate with no parameters, but not a lambda expression with no parameters?
ThreadPool.QueueUserWorkItemhas a specific delegate in its signature; Invoke just hasDelegate. Lambda expressions and anonymous methods can only be converted to a specific delegate type.It’s just a bad error message. It means, “I don’t know exactly which delegate type you’re trying to convert to.”
You’re using an anonymous method without a parameter list at all which can be converted to any delegate type which doesn’t use out/ref parameters. If you tried
delegate() { ... }(i.e. an explicitly empty parameter list) then it wouldn’t work. This “I don’t care about parameters” ability of anonymous methods is the only feature they have which lambda expressions don’t.It’s easiest to demonstrate all of this in the context of simple assignments, IMO: