C# 委托知识总结

Posted Rennix

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了C# 委托知识总结相关的知识,希望对你有一定的参考价值。

1.什么是委托,为什么要使用委托

我正在埋头苦写程序,突然想喝水,但是又不想自己去掉杯水而打断自己的思路,于是我就想让女朋友去给我倒水。她去给我倒水,首先我得让她知道我想让她干什么,通知她之后我可以继续写自己的程序,而倒水的工作就交给了她。这样的过程就相当于一个委托。

在程序过程中,当程序正在处理某个事件的时候,我需要另外的程序代码去辅助处理一些事情,于是委托另一个程序模块去处理,而委托就可以达到这种目的,我可以利用委托通知另外的程序模块,该去调用哪个函数方法。委托其实就起到了这样一个作用,将函数签名传递到了另一个函数中。或许这样讲还是有些模糊,看看后面的具体实例。

 

2.委托的定义

delegate int Add(int num1,int num2);

delegate void ConvertNum(string result);

上面是定义两个委托的例子,其实很简单。声明一个委托使用delegate关键字,上面分别是定义的带返回值的委托和不带返回值的委托, 

两个委托都有传递参数,当然也可以不传递参数。其实委托也是一个类,委托派生为System.MulticastDelegate,而System.MulticastDelegate

又继承System.Delegate,如果你知道这个也就明白委托其实是一个特殊的类。

复制代码
 1 public delegate string TeaDelegate(string spText);
 2 
 3     public class DelegateSource
 4     {
 5         public void TestDelegate()
 6         {
 7             Operator op = new Operator();
 8             TeaDelegate tea = new TeaDelegate(op.GetTea);
 9             Console.WriteLine("去给我倒杯水");
10             Console.WriteLine();
11             string result=tea("去给我倒杯水");
12             Thread.Sleep(5000);
13             Console.WriteLine(result);
14             Console.WriteLine();
15         }
16     }
17 
18     public class Operator
19     {
20         /// <summary>
21         /// 确定是否还有水
22         /// </summary>
23         private bool flag = true;
24 
25         public string GetTea(string spText)
26         {
27             if (spText == "去给我倒杯水")
28             {
29                 if (flag)
30                 {
31                     return "老公,茶来了";
32                 }
33                 else
34                 {
35                     return "老公,没有水了";
36                 }
37             }
38             return "等待.......";
39         }
40     }
复制代码

 

 

输出结果

 

上面使用最普通的一种方式来定义了一个委托的使用,这个例子虽然很简单,但是能够很形象的描述委托的使用。

 

3.委托的三种形式

(1).推断

复制代码
 1 public delegate string TeaDelegate(string spText);
 2 
 3     public class DelegateSource
 4     {
 5         public void TestDelegate()
 6         {
 7             Operator op = new Operator();
 8             TeaDelegate tea = op.GetTea;
 9             Console.WriteLine("去给我倒杯水");
10             Console.WriteLine();
11             string result=tea("去给我倒杯水");
12             Thread.Sleep(5000);
13             Console.WriteLine(result);
14             Console.WriteLine();
15         }
16     }
17 
18     public class Operator
19     {
20         /// <summary>
21         /// 确定是否还有水
22         /// </summary>
23         private bool flag = true;
24 
25         public string GetTea(string spText)
26         {
27             if (spText == "去给我倒杯水")
28             {
29                 if (flag)
30                 {
31                     return "老公,茶来了";
32                 }
33                 else
34                 {
35                     return "老公,没有水了";
36                 }
37             }
38             return "等待.......";
39         }
40     }
复制代码

 

在委托定义的例子中我们看到委托的使用方法是在委托实例化的时候指定的[new DelegateName(FunctionName)],这里可能表述不是太但是代码应该看得白了。 而委托的推断,并没有new 委托这个步骤,而是直接将Function 指定给委托。

(2).匿名函数

复制代码
 1 public delegate string TeaDelegate(string spText);
 2 
 3     public class DelegateSource
 4     {
 5         public void TestDelegate()
 6         {
 7             Operator op = new Operator();
 8             bool flag = true;
 9             TeaDelegate tea = delegate(string spText)
10             {
11                 if (spText == "去给我倒杯水")
12                 {
13                     if (flag)
14                     {
15                         return "老公,茶来了";
16                     }
17                     else
18                     {
19                         return "老公,没有水了";
20                     }
21                 }
22                 return "等待.......";
23             };
24 
25             Console.WriteLine("去给我倒杯水");
26             Console.WriteLine();
27             string result=tea("去给我倒杯水");
28             Thread.Sleep(5000);
29             Console.WriteLine(result);
30             Console.WriteLine();
31         }
32     }
复制代码

 

至于匿名委托,给人的感觉更为直接了,都不用显示的指定方法名,因为根本没有方法,而是指定的匿名方法。匿名方法在.NET 中提高了 

代码的可读性和优雅性。对于更多操作较少的方法直接写为匿名函数,这样会大大提高代码的可读性。这里有两个值得注意的地方: 第一,不能使用

跳转语句跳转到该匿名方法外,第二 不能使用ref,out修饰的参数

(3).多播委托

复制代码
 1 public delegate string TeaDelegate(string spText);
 2 
 3     public class DelegateSource
 4     {
 5         public void TestDelegate()
 6         {
 7             Operator op = new Operator();
 8 
 9             TeaDelegate tea1 = op.GetTea;
10             TeaDelegate tea2 = op.Speak;
11             TeaDelegate tea = tea1 + tea2;
12 
13             Console.WriteLine("去给我倒杯水");
14             Console.WriteLine();
15             string result=tea("去给我倒杯水");
16             Thread.Sleep(5000);
17             Console.WriteLine(result);
18             Console.WriteLine();
19         }
20     }
21 
22     public class Operator
23     {
24         /// <summary>
25         /// 确定是否还有水
26         /// </summary>
27         private bool flag = true;
28 
29         public string GetTea(string spText)
30         {
31             if (spText == "去给我倒杯水")
32             {
33                 if (flag)
34                 {
35                     return "老公,茶来了";
36                 }
37                 else
38                 {
39                     return "老公,没有水了";
40                 }
41             }
42             return "等待.......";
43         }
44 
45 
46         public string Speak(string spText)47         {48             Console.WriteLine("\\n去把我的设计图稿拿来");49             return null;50         }51     }
复制代码

 

  还是上面的那个实例,我不尽想让女朋友去给我掉杯水,还让她帮我将程序设计图稿拿过来。这个时候做的就不是一件事了,而是多件。

程序中也有很多这种情况,于是我们需要多播委托,在一个委托上指定多个执行方法,这是在程序中可以行的。上面提到了,委托直接继承于

System.MulticastDelegate,正是因为这个类可以实现多播委托。如果调用多播委托,就可以按顺序连续调用多个方法。为此,委托的签名就必须返回void;否则,就只能得到委托调用的最后一个方法的结果。所以在上面的这段代码中是得不到结果的

 

4.事件

使用C#编程,无论是WinForm,WebForm 给人很难忘得就是它的控件,而他们的控件库使用方式都是使用使用事件驱动模式,而事件驱动模式却少不了委托。话不多说,看代码能够更清好的理解事件和委托之间的联系. 

复制代码
 1 public delegate void MyDelegate(string name);
 2 
 3     public class EventSource
 4     {
 5         public event MyDelegate Event_Delegate;
 6 
 7         public void SetCustomer(string name)
 8         {
 9             Console.WriteLine("事件发生.....\\n");
10             Console.WriteLine("hi! "+name);
11         }
12 
13         public void TestEvent()
14         {
15             EventSource source = new EventSource();
16             Console.WriteLine("订阅事件.....\\n");
17             source.Event_Delegate += new MyDelegate(source.SetCustomer);
18             Console.WriteLine("触发事件.....\\n");
19             source.Event_Delegate("hechen");
20             Console.WriteLine("..................");
21         }
22     }