浅谈C# 中的委托和事件

2019-12-30 15:06:43丽君

如你所见,委托GreetingDelegate出现的位置与 string相同,string是一个类型,那么GreetingDelegate应该也是一个类型,或者叫类(Class)。但是委托的声明方式和类却完全不同,这是怎么一回事?实际上,委托在编译的时候确实会编译成类。因为Delegate是一个类,所以在任何可以声明类的地方都可以声明委托。更多的内容将在下面讲述,现在,请看看这个范例的完整代码:


using System;
 using System.Collections.Generic;
 using System.Text;

 namespace Delegate {
   //定义委托,它定义了可以代表的方法的类型
   public delegate void GreetingDelegate(string name);
     class Program {

      private static void EnglishGreeting(string name) {
        Console.WriteLine("Morning, " + name);
      }

      private static void ChineseGreeting(string name) {
        Console.WriteLine("早上好, " + name);
      }

      //注意此方法,它接受一个GreetingDelegate类型的方法作为参数
      private static void GreetPeople(string name, GreetingDelegate MakeGreeting) {
        MakeGreeting(name);
       }

      static void Main(string[] args) {
        GreetPeople("Jimmy Zhang", EnglishGreeting);
        GreetPeople("张子阳", ChineseGreeting);
        Console.ReadKey();
      }
     }
   }

输出如下:

Morning, Jimmy Zhang
早上好, 张子阳

我们现在对委托做一个总结:

委托是一个类,它定义了方法的类型,使得可以将方法当作另一个方法的参数来进行传递,这种将方法动态地赋给参数的做法,可以避免在程序中大量使用If-Else(Switch)语句,同时使得程序具有更好的可扩展性。

将方法绑定到委托

看到这里,是不是有那么点如梦初醒的感觉?于是,你是不是在想:在上面的例子中,我不一定要直接在GreetPeople()方法中给 name参数赋值,我可以像这样使用变量:


static void Main(string[] args) {
   string name1, name2;
   name1 = "Jimmy Zhang";
   name2 = "张子阳"; 

   GreetPeople(name1, EnglishGreeting);
   GreetPeople(name2, ChineseGreeting);
   Console.ReadKey();
 }

而既然委托GreetingDelegate 和 类型 string 的地位一样,都是定义了一种参数类型,那么,我是不是也可以这么使用委托?


static void Main(string[] args) {
   GreetingDelegate delegate1, delegate2;
   delegate1 = EnglishGreeting;
   delegate2 = ChineseGreeting;

   GreetPeople("Jimmy Zhang", delegate1);
     GreetPeople("张子阳", delegate2);
     Console.ReadKey();
 }

如你所料,这样是没有问题的,程序一如预料的那样输出。这里,我想说的是委托不同于string的一个特性:可以将多个方法赋给同一个委托,或者叫将多个方法绑定到同一个委托,当调用这个委托的时候,将依次调用其所绑定的方法。在这个例子中,语法如下: