委托/ lambda打字和强制如何工作?

时间:2010-09-29 14:03:13

标签: c# delegates lambda type-inference anonymous-methods

我注意到在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() {
        // ...
    }
}

那是很多例子。我想我的问题如下:

  1. 为什么Invoke总是拒绝lambda函数或匿名委托,但ThreadPool.QueueUserWorkItem还不错?

  2. 究竟是什么“无法将匿名方法转换为'System.Delegate',因为它不是委托类型”是指“无论如何?”

  3. 为什么ThreadPool.QueueUserWorkItem接受没有参数的匿名委托,而不接受没有参数的lambda表达式?

1 个答案:

答案 0 :(得分:9)

  1. ThreadPool.QueueUserWorkItem在其签名中有特定的委托;调用只有Delegate。 Lambda表达式和匿名方法只能转换为特定的委托类型。

  2. 这只是一个错误的错误消息。这意味着,“我不确切地知道您要转换为哪种委托类型。”

  3. 您使用的是匿名方法,而且没有参数列表,可以将其转换为不使用out / ref参数的任何委托类型。如果您尝试delegate() { ... }(即显式空参数列表),那么它将无效。匿名方法的“我不关心参数”能力是他们拥有的唯一特性,而lambda表达式则没有。

  4. 最容易在简单分配的背景下展示所有这些,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"); };