- 接着在控制台中调用这两个方法
public delegate int DemoDelegate(int num); class Program { static void Main(string[] args) { Demo demo = new Demo(); DemoDelegate dd = demo.NumAdd; dd(2); Console.WriteLine("方法所属实例:"+dd.Target); Console.WriteLine("调用方法:"+dd.Method); DemoDelegate staticDd = Demo.Num; staticDd(2); Console.WriteLine("方法所属实例:" + staticDd.Target); Console.WriteLine("调用方法:" + staticDd.Method); Console.ReadLine(); } }
运行以上代码,输出结果如下:
我们可以看到,将静态方法赋值给委托对象后打印方法所属实例为空。
三、泛型委托类型
在一些情况下我们不确定参数类型和返回值类型,这时我们就需要用到泛型委托类型,语法如下:
public delegate T DemoDelegate<T>(T arg);
我们具体看一下例子:
public delegate T DemoDelegate<T>(T num); class Demo { public int NumAdd(int num) { return ++num; } } class Program { static void Main(string[] args) { Demo demo = new Demo(); DemoDelegate<int> dd = demo.NumAdd; Console.WriteLine(dd(2)); Console.ReadLine(); } }
运行上面的代码,控制台将会输出结果 3
注意:我们可以将返回值类型或者参数类型固定,例如:
public delegate string DemoDelegate<T>(T arg); public delegate T DemoDelegate<T>(int arg);
使用泛型委托的好处是可以写出一组委托类型,这组方法可以拥有热议类型的返回值和任意数量的参数。下一小节我们就来看一下具体怎么用。
四、Action 和 Func
1.Func
Func是一个具有返回类型的方法,它的类型参数数量可以多达16个,其中包括0到多个输入类型参数和一个输出类型参数。下面的代码段展示了Func部分类型参数:
delegate TResult Func<out TResult>(); delegate TResult Func<in T,out TResult>(T arg); delegate TResult Func<in T1,in T0,out TResult>(T t1,T t2);
2. Action
Action 是一个不具有返回类型的方法,他的类型参数数量同样多达16个。下面展示了部分Action类型参数:
delegate void Action(); delegate void Action<in T> (T t); delegate void Action(in T1,in T2)(T t1,T t2);
我们来看一下例子,以Func为例,Action同理
class Demo { public void Num<T>(T[] array, Func<T, T> func) { for (int i = 0; i < array.Length; i++) { Console.WriteLine(func(array[i])); } } } class Program { static void Main(string[] args) { Demo demo = new Demo(); int[] array = new int[] {2,4,6 }; demo.Num<int>(array, NumAdd); Console.ReadLine(); } static int NumAdd(int num) { return ++num; } }
从代码中可以看出,我们将 Demo 类中的 Num 方法的第二个参数类型写成了 Func<T,T>,这里的意思是委托实例的返回类型和类型参数都是T类型。我们在Main函数中通过委托,控制台输出结果是 3、5、7 。这时我们就看出了使用 Func 和 Action 的优点了,我们不需要在外部显式的定义委托,比较方便。
五、冷知识
1.委托与接口
一般来说接口可以解决的问题,委托同样也可以解决,那么什么时候使用委托呢?我们来看一下:
- 当需要多播的时候;
- 订阅者需要多次实现接口的时候。
2.委托兼容性
- 委托类型
委托类型之间互不兼容,即使它们的签名一样也不行,也就是说如下的写法是错误的。
delegate void DD1(); delegate void DD2(); DD1 dd1=Method; DD2 dd2=dd1;
- 委托实例
如果委托实例具有相同的方法目标,那么委托实例就是相等的。
- 参数
当调用一个方法时,提供的参数可以比方法参数更具体。例如被调用的方法参数是 Object 类型,但是提供的参数是 String 类型,这时程序不会报错,因为string 来自 object,string 比 object 更具体。(委托只支持引用转换)
- 返回值
同参数一样,当调用方法时,可以获得一个比被调用方法返回值更具体的返回值。
delegate void DD1(); delegate void DD2(); DD1 dd1=Method; DD2 dd2=dd1;
- 委托实例
如果委托实例具有相同的方法目标,那么委托实例就是相等的。
- 参数
当调用一个方法时,提供的参数可以比方法参数更具体。例如被调用的方法参数是 Object 类型,但是提供的参数是 String 类型,这时程序不会报错,因为string 来自 object,string 比 object 更具体。(委托只支持引用转换)
- 返回值
同参数一样,当调用方法时,可以获得一个比被调用方法返回值更具体的返回值。