日期:2014-05-17  浏览次数:20782 次

[C#] 委托与方法组

在编写委托相关代码,尤其是泛型委托相关代码的时候,经常会遇到下面这种问题:


定义一个委托类型:

public delegate void TestDelegate<T>(T arg);

定义一个测试用的类,及方法:

public class TestClass
{
    public void TestFunc4Delegate(int arg){}
    public void TestFunc1(Delegate dele){}
    public void TestFunc2<T>(TestDelegate<T> dele){}
}

测试用的代码:

1.    TestClass tc = new TestClass();
2.    tc.TestFunc1(tc.TestFunc4Delegate);
3.    tc.TestFunc2(tc.TestFunc4Delegate);

然后会发现,上面的测试用代码有编译错误,其中第二行的编译错误是“无法从方法组转换为System.Delegate”,第三行的编译错误是“无法从用法中推导出TestClass.TestFunc2<T>(TestDelegate<T>)的类型实参”。

以上报错的核心问题就在“方法组”上。“方法组”是一个编译器概念,由于C#支持方法重载,所以名字相同的方法可能会有多种重载实现方式,这所有的名称相同的重载方法合在一起就成为一个方法组。上面的例子中tc.TestFunc4Delegate实际上指向的是名为Test4Delegate的方法组,尽管上面的类定义中并没有对Test4Delegate做任何重载,它依然是一个方法组。

既然是方法组,那么问题就来了——无法确定需要的到底是方法组中的哪一个重载。正因为确定不了,所以编译器无法决定用什么委托子类型来创建委托实例,同样也就无法推导出泛型的正确类型。

想要让上例中的调用可以通过编译,就必须指定限定类型,如这样:

tc.TestFunc1((TestDelegate<int>)tc.TestFunc4Delegate);
tc.TestFunc2<int>(tc.Test4Delegate);