引言

Delegate是Dotnet1.0的时候已经存在的特性了,但由于在实际工作中一直没有机会使用Delegate这个特性,所以一直没有对它作整理。这两天,我再度翻阅了一些关于Delegate的资料,并开始正式整理这个C#中著名的特性。本文将由浅入深的谈一下Delegate这个特性。

一.Delegate是什么?

Delegate中文翻译为“委托”。Msdn中对Delegate的解释如下:

C#中的委托类似于C或C++中的函数指针。使用委托使程序员可以将方法引用封装在委托对象内。然后可以将该委托对象传递给可调用所引用方法的代码,而不必在编译时知道将调用哪个方法。与C或C++中的函数指针不同,委托是面向对象、类型安全的,并且是安全的。

如果你是第一次接触Delegate这个概念,你可能会对上面这段文字感觉不知所云,不过不要紧,你可以先把Delegate认为就是一个函数指针。

而当你面对一个虚无的概念时,最好的应对方法就是直接看实例。下面一个简单的Delegate使用例子。

  1. class Program
  2. {
  3. static void OtherClassMethod(){
  4. Console.WriteLine("Delegate an other class's method");
  5. }
  6.  
  7. static void Main(string[] args)
  8. {
  9. var test = new TestDelegate();
  10. test.delegateMethod = new TestDelegate.DelegateMethod(test.NonStaticMethod);
  11. test.delegateMethod += new TestDelegate.DelegateMethod(TestDelegate.StaticMethod);
  12. test.delegateMethod += Program.OtherClassMethod;
  13. test.RunDelegateMethods();
  14. }
  15. }
  16.  
  17. class TestDelegate
  18. {
  19. public delegate void DelegateMethod(); //声明了一个Delegate Type
  20.  
  21. public DelegateMethod delegateMethod; //声明了一个Delegate对象
  22.  
  23. public static void StaticMethod()
  24. {
  25. Console.WriteLine("Delegate a static method");
  26. }
  27.  
  28. public void NonStaticMethod()
  29. {
  30. Console.WriteLine("Delegate a non-static method");
  31. }
  32.  
  33. public void RunDelegateMethods()
  34. {
  35. if(delegateMethod != null){
  36. Console.WriteLine("---------");
  37. delegateMethod.Invoke();
  1. Console.WriteLine("---------");
  2. }
  3. }
  4. }

上面是一个Delegate的使用例子,运行看看结果吧。下面我稍微解释一下:

【1】public delegate void DelegateMethod();这里声明了一个Delegate的类型,名为DelegateMethod,这种Delegate类型可以搭载:返回值为void,无传入参数的函数。

【2】public DelegateMethod delegateMethod;这里声明了一个DelegateMethod的对象(即,声明了某种Delegate类型的对象)。

区分:DelegateMethod是类型,delegateMethod是对象。

【3】为什么上面说Delegate可以看做是函数指针呢?看下面这段代码:

  1. test.delegateMethod = new TestDelegate.DelegateMethod(test.NonStaticMethod);
  2. test.delegateMethod += new TestDelegate.DelegateMethod(TestDelegate.StaticMethod);
  3. test.delegateMethod += Program.OtherClassMethod;

这里delegateMethod搭载了3个函数,而且可以通过调用delegateMethod.Invoke();运行被搭载的函数。这就是Delegate可以看作为函数指针的原因。上面这段代码中,delegateMethod只能搭载:返回值为void,无传入参数的函数(见:NonStaticMethod,StaticMethod,OtherClassMethod的定义),这和Delegate类型声明有关(见DelegateMethod的声明:public delegate void DelegateMethod())。

【4】Delegate在搭载多个方法时,可以通过+=增加搭载的函数,也可以通过-=来去掉Delegate中的某个函数。

二.Delegate和C++中函数指针的区别

Delegate和C++中的函数指针很像,但如果深入对比,发现其实还是有区别的,区别主要有三个方面(参考Stanley B. Lippman的一篇文章)

1) 一个 delegate对象一次可以搭载多个方法(methods),而不是一次一个。当我们唤起一个搭载了多个方法(methods)的delegate,所有方法以其“被搭载到delegate对象的顺序”被依次唤起。

2) 一个delegate对象所搭载的方法(methods)并不需要属于同一个类别。一个delegate对象所搭载的所有方法(methods)必须具有相同的原型和形式。然而,这些方法(methods)可以即有static也有non-static,可以由一个或多个不同类别的成员组成。

3) 一个delegate type的声明在本质上是创建了一个新的subtype instance,该 subtype 派生自 .NET library framework 的 abstract base classes Delegate 或 MulticastDelegate,它们提供一组public methods用以询访delegate对象或其搭载的方法(methods) ,与函数指针不同,委托是面向对象、类型安全并且安全的。

看完上面关于Delegate的介绍,相信大家对它也有所了解了,下面我们将进行更深入地讨论!

三.Delegate什么时候该用?

看完上面的介绍,你可以会有一些疑问,为什么会有Delegate?实际中什么时候会用到?什么时候应该去用? 在回答这些问题之前,大家可以先看看下面这段代码:

  1. class Program
  2. {
  3. static void Main(string[] args)
  4. {
  5. var car = new Car(15);
  6. new Alerter(car);
  7. car.Run(120);
  8. }
  9. }
  10.  
  11. class Car
  12. {
  13. public delegate void Notify(int value);
  14. public event Notify notifier;
  15.  
  16. private int petrol = 0;
  17. public int Petrol
  18. {
  19. get { return petrol; }
  20. set
  21. {
  22. petrol = value;
  23. if (petrol < 10) //当petrol的值小于10时,出发警报
  24. {
  25. if (notifier != null)
  26. {
  27. notifier.Invoke(Petrol);
  28. }
  29. }
  30. }
  31. }
  32.  
  33. public Car(int petrol)
  34. {
  35. Petrol = petrol;
  36. }
  37.  
  38. public void Run(int speed)
  39. {
  40. int distance = 0;
  41. while (Petrol > 0)
  42. {
  43. Thread.Sleep(500);
  44. Petrol--;
  45. distance += speed;
  46. Console.WriteLine("Car is running... Distance is " + distance.ToString());
  47. }
  48. }
  49. }
  50.  
  51. class Alerter
  52. {
  53. public Alerter(Car car)
  54. {
  55. car.notifier += new Car.Notify(NotEnoughPetrol);
  56. }
  57.  
  58. public void NotEnoughPetrol(int value)
  59. {
  60. Console.ForegroundColor = ConsoleColor.Red;
  61. Console.WriteLine("You only have " + value.ToString() + " gallon petrol left!");
  62. Console.ResetColor();
  63. }
  64. }

看完了上面的代码后,你可能会问:为什么不在public int Petrol中直接调用Alerter.NotEnoughPetrol呢?因为Car模块和Alerter模块本身是两个独立的子系统,如果直接调用,耦合性就会增加,这不是我们愿意看到的。

其实以上的代码是设计模式中的观察者模式(观察者模式又称Source/Listener模式)的实现,当汽车在运行中汽油量<10时,警报器便会发出警报。在上面代码中,Delegate相当于一个存放回调函数的函数指针,使用Delegate,我们可以非常方便地实现观察者模式。而其实,在需要使用回调函数时,我们都可以考虑使用Delegate。

不知道你有没有发现在上面的代码中还有一个问题呢?

  1. public event Notify notifier;

上面的代码中,我们定义了一个Event,而事实上:

  1. public Notify notifier;

这样写,也完全可以满足我们的需求,这就引出了我们的另一个问题,Delegate和Event!

四.Delegate与Event

【1】Delegate和Event的关系

看微软的代码时,我们会发现Delegate和Event这两个关键字经常会一起出现!究竟他们是什么关系呢?

Msdn中,有一段话描述Delegate和Event之间的关系,其实很简单:

        声明事件:若要在类内声明事件,首先必须声明该事件的委托类型。

【2】Delegate和Event配合使用的效果

看下面几幅图,这是我从一个C#的Application程序截下来的:

从上图看到,在响应图形界面的操作中,我们用到了Event和Delegate,相信这也我们使用Event和Delegate最频繁的地方了。这里我还想罗嗦一下,平时需要我们自己写代码的界面事件响应函数,如:button_Click(…),其实都是回调函数,在自动生成的文件Form1.Designer.cs中,VS把事件和其对应的回调函数(即:button_Click(…)等)关联起来,当触发某事件时,对应的回调函数便会执行。

【3】“public Notify notifier”和“public event Notify notifier”的区别

关于这个问题,我们直接ildasm看看IL代码吧:>

“public Notify notifier”的IL代码,如图:

“public event Notify notifier”的IL代码,如图:

差别其实已经很明显了,“public Notify notifier”相当于Class里面的Field,访问级别是public,而“public event Notify notifier”则相当于Property,访问级别是private!由于以上的差别,他们在某些使用上,会稍有不同,详细的可参考shensr写的《delegate vs. event》。

五.Delegate中的Invoke与BeginInvoke方法

简单说一下,Invoke与BeginInvoke都是执行Delegate里的搭载函数,而不同的是:Invoke是一个同步方法,BeginInvoke是一个异步方法。关于这个,有一篇文章《Invoke and BeginInvoke》,对此介绍的比较详细,这里就不多说了。

六.小结

回顾一下,到底什么时候我们可能会用到Delegate:

【1】.当我们在C#中需要类似函数指针这样东西时。

【2】.当我们需要使用回调函数的时候。

【3】.需要异步调用的时候。

【4】.实现观察者模式的时候。

【5】.处理事件响应的时候。

以上内容均为个人看法,如果有错漏,请各位及时指出:>

转载请说明出处,谢谢![hyddd(http://www.cnblogs.com/hyddd/)]

参考资料

【1】Msdn委托教程Msdn事件教程

【2】《深入探索面向对象事件(Delegate)机制

【3】《对.net事件的看法

【4】《Invoke and BeginInvoke

【5】《delegate vs. event

【6】《C#事件(event)解析

【7】《C# Delegate 简介

谈C#中的Delegate的更多相关文章

  1. 浅谈c#中的delegate和event了

    一.开篇忏悔 对自己最拿手的编程语言C#,我想对你说声对不起,因为我到现在为止才明白c#中的delegate和event是怎么用的,惭愧那.好了,那就趁着阳光明媚的早晨简单来谈谈delegate和ev ...

  2. C#中的Delegate

    谈C#中的Delegate http://www.cnblogs.com/hyddd/archive/2009/07/26/1531538.html

  3. [转] C#中的delegate 和 event

    转至:here 终于会用c#中的delegate(委托) 作者:qq826364410 引言 Delegate是Dotnet1.0的时候已经存在的特性了,但由于在实际工作中一直没有机会使用Delega ...

  4. 【分析】浅谈C#中Control的Invoke与BeginInvoke在主副线程中的执行顺序和区别(SamWang)

    [分析]浅谈C#中Control的Invoke与BeginInvoke在主副线程中的执行顺序和区别(SamWang) 今天无意中看到有关Invoke和BeginInvoke的一些资料,不太清楚它们之间 ...

  5. 浅谈Java中的equals和==(转)

    浅谈Java中的equals和== 在初学Java时,可能会经常碰到下面的代码: 1 String str1 = new String("hello"); 2 String str ...

  6. jquery中on/delegate的原理

    jquery中on/delegate的原理 早期版本中叫delegate, 后来有过live函数,再后来统一用on.下面的方法等效: // jQuery 1.3 $(selector).(events ...

  7. 由项目浅谈JS中MVVM模式

    文章版权由作者李晓晖和博客园共有,若转载请于明显处标明出处:http://www.cnblogs.com/naaoveGIS/. 1.    背景 最近项目原因使用了durandal.js和knock ...

  8. 浅谈Linux中的信号处理机制(二)

    首先谢谢 @小尧弟 这位朋友对我昨天夜里写的一篇<浅谈Linux中的信号处理机制(一)>的指正,之前的题目我用的“浅析”一词,给人一种要剖析内核的感觉.本人自知功力不够,尚且不能对着Lin ...

  9. 浅谈Java中的对象和引用

    浅谈Java中的对象和对象引用 在Java中,有一组名词经常一起出现,它们就是“对象和对象引用”,很多朋友在初学Java的时候可能经常会混淆这2个概念,觉得它们是一回事,事实上则不然.今天我们就来一起 ...

随机推荐

  1. ubuntu 错误 & 解决

    1.ssh时出现“段错误(核心已转储)” 原因:说明与ssh有关的内核代码被修改过并且部分代码访问内存过界 解决:1.将内核代码被修改过的部分修改回来        2.sudo apt-get re ...

  2. 在适配iPhone 6 Plus屏幕的时候,模拟器上两边有很细的白边如何解决

    取消掉Constrain to margin 然后添加左右约束 版权声明:本文为博主原创文章,未经博主允许不得转载.    

  3. openstack neutron网络主机节点网口配置 liberty版本之前的

  4. 第三百五十五天 how can I 坚持

    快一年了,三百五十五天了,等写个程序算算时间,看看日期和天数能不能对的上,哈哈. 计划还是未制定,天气预报还是没有写完,立马行动,发完这个博客,立马行动. 计划:设计模式1个月,三大框架3个月,计算机 ...

  5. [翻译]创建ASP.NET WebApi RESTful 服务(8)

    本章讨论创建安全的WebApi服务,到目前为止,我们实现的API都是基于未加密的HTTP协议,大家都知道在Web中传递身份信息必须通过HTTPS,接下来我们来实现这一过程. 使用HTTPS 其实可以通 ...

  6. MYSQL数据库性能调优之二:定位慢查询

    windows下开启慢查询: 第一步:先查看版本 第二步查看查询日志和慢查询配置 第三步:配置开启慢查询 在my.ini配置文件的[mysqld]选项下增加: slow_query_log=TRUE ...

  7. How to run a terminal inside of vim?

    [How to run a terminal inside of vim?] :sh turn vim into shell mode d+trl back to vim 参考:http://stac ...

  8. ASP.NET前台AJAX方法调用后台的方法写法

    前台: <input id="AjaxDemo" type="button" onclick="get()" value=" ...

  9. VS2008 工程中部分文件不参与编译 从生成中排除【Worldsing笔记】

    Visual Studio 2008 .VS2008.VC2008工程源文件配置.编译配置   有时编写代码时,往往存在这样的需求(或是希望有这样的功能):一个工程经过不共同的配置实现不同的版本或是功 ...

  10. How Tomcat Works(十四)补充

    在How Tomcat Works(十四)中,本人并没有对javax.servlet.Filter及javax.servlet.FilterChain做详细的描述,本文在这里做一下补充 FilterC ...