各版本.NET委托的写法回顾

开发 后端
本文讲述了各版本.NET委托写法的变化。当然,委托本身其实从来没有改变过,改变的一直都是委托的“写法”。

在《关于最近面试的一点感想》一文中,Michael同学谈到他在面试时询问对方“delegate在.net framework1.1,2.0,3.5各可以怎么写”这个问题。于是乎,有朋友回复道“请问楼主,茴香豆的茴有几种写法”,“当代孔乙己”,独乐,众乐。看了所有的评论,除了某些朋友认为“的确不该不知道这个问题”之外,似乎没有什么人在明确支持楼主。

不过我支持,为什么?因为我也提过出这样的问题:各版本.NET委托的写法有何不同?

这样,我们暂且不提应聘“高级开发人员”的人,在“自称熟悉各版本.NET框架”的前提下,是否应该知道这个答案。我们也暂且不提Michael同学提问的“目的”是什么。老赵就先单独针对这个问题进行解释,然后谈谈自己为什么会提出这个问题吧。

可能有一件事情需要说在前面,那就是:委托本身其实从来没有改变过,改变的一直都是委托的“写法”。因此更确切地说,改变的只是“编译器”。而本文所有内容都用C#来实现,其实谈得也都是C#编译器本身——但是其实VB.NET也有变化埃再由于.NET版本和C#版本的关系也是非常密切的,因此全文就使用.NET版本进行指代了。

.NET 1.x中委托的写法

委托,如果不追究细节,从表面上来看我们可以将其通俗地理解为一个安全的“函数指针”。当然,这个函数指针其实也是一个对象,有自己的成员,也会封装了被调用方的上下文等等。至于委托的定义和使用方式,则是这样的:

  1. public delegate int SomeDelegate(string arg1, bool arg2);  
  2.  
  3. public static int SomeMethod(string arg1, bool arg2) { return 0; }  
  4.  
  5. public class SomeClass  
  6. {  
  7.     public int SomeMethod(string a1, bool a2) { return 0; }  
  8.  
  9.     public event SomeDelegate SomeEvent;  
  10. }  
  11.  
  12. static void Main(string[] args)  
  13. {  
  14.     SomeClass someClass = new SomeClass();  
  15.     SomeDelegate someDelegate = new SomeDelegate(someClass.SomeMethod);  
  16.  
  17.     someClass.SomeEvent += new SomeDelegate(SomeMethod);  
  18. }  
  19.  

可见,在.NET 1.x中需要使用new DelegateType(...)的方式来创建一个委托对象。不过,作为委托对象内部的方法它既可以是实例方法,也可以是静态方法。此外,方法只需要匹配委托类型的签名和返回值即可,方法参数的名称不会成为约束。

嗯,就是这么简单。

.NET 2.0中委托的写法

.NET委托引入了范型,且写法略有简化:

  1. public delegate TResult MyFunc(T1 a1, T2 a2);  
  2.  
  3. public static int SomeMethod(string a1, bool a2) { return 0; }  
  4.  
  5. static void Main(string[] args)  
  6. {  
  7.     MyFunc<stringboolint> myFunc = SomeMethod;  
  8. }  
  9.  

在.NET 2.0中,new DelegateType已经可以省略,开发人员可以直接将方法赋值给一个委托对象的引用。当然,这个改进不值一提,.NET 2.0中委托写法的关键在于引入了“匿名方法”:

  1. public static void TestRequest(string url)  
  2. {  
  3.     WebRequest request = HttpWebRequest.Create(url);  
  4.     request.BeginGetResponse(delegate(IAsyncResult ar)  
  5.     {  
  6.         using (WebResponse response = request.EndGetResponse(ar))  
  7.         {  
  8.             Console.WriteLine("{0}: {1}", url, response.ContentLength);  
  9.         }  
  10.     },  
  11.     null);  
  12. }  

匿名方法,简单地说就是内联在方法内部的委托对象,它的关键便在于形成了一个闭包(委托执行时所需的上下文)。如上面的代码中,BeginGetResponse的***个参数(委托)可以直接使用TestRequest方法的参数url,以及方法内的“局部”变量request。如果没有匿名函数这个特性的话,代码写起来就麻烦了,例如在.NET 1.x中您可能就必须这么写:

  1. public static void TestRequest(string url)  
  2. {  
  3.     WebRequest request = HttpWebRequest.Create(url);  
  4.     object[] context = new object[] { url, request };  
  5.     request.BeginGetResponse(TestAsyncCallback, context);  
  6. }  
  7.  
  8. public static void TestAsyncCallback(IAsyncResult ar)  
  9. {   
  10.     object[] context = (object[])ar.AsyncState;  
  11.     string url = (string)context[0];  
  12.     WebRequest request = (WebRequest)context[1];  
  13.  
  14.     using (WebResponse response = request.EndGetResponse(ar))  
  15.     {  
  16.         Console.WriteLine("{0}: {1}", url, response.ContentLength);  
  17.     }  
  18. }  
  19.  

此时,我们往往会发现,开发人员需要花费大量的精力,为一小部分代码维护一大段上下文。例如在这段代码中,我们会将url和request对象塞入一个object数组中,在回调函数中再通过危险的Cast操作恢复数据。如果您希望“强类型”,那么只能为每个回调创建一个新的上下文对象,维护起来可能更加麻烦——要知道,在并行编程,异步调用越来越重要的今天,如果没有匿名方法自动保留上下文的特性,开发人员会为这些“额外工作”疲于奔命的。

可能您会说,匿名方法的可读性不佳,因为需要“内联”。一个方法中内联太多,维护成本就上去了,所以匿名方法并不推荐使用。我想说的是,您错了。如果为了可维护性,要将方法独立拆开,也可以利用匿名方法的优势:

  1. public static void TestRequest(string url)  
  2. {  
  3.     WebRequest request = HttpWebRequest.Create(url);  
  4.     request.BeginGetResponse(delegate(IAsyncResult ar)  
  5.     {  
  6.         TestAsyncCallback(ar, request, url);  
  7.     }, null);  
  8. }  
  9.  
  10. public static void TestAsyncCallback(IAsyncResult ar, WebRequest request, string url)  
  11. {  
  12.     using (WebResponse response = request.EndGetResponse(ar))  
  13.     {  
  14.         Console.WriteLine("{0}: {1}", url, response.ContentLength);  
  15.     }  
  16. }  
  17.  

如果借助.NET 3.5中的Lambda表达式,代码可以写的更简单易读:

  1. public static void TestRequest(string url)  
  2. {  
  3.     WebRequest request = HttpWebRequest.Create(url);  
  4.     request.BeginGetResponse(ar => TestAsyncCallback(ar, request, url), null);  
  5. }  
  6.  

以上就总结了各版本.NET委托的写法。

【编辑推荐】

  1. C# Actor模型开发实例:网络爬虫
  2. 强类型和Actor:ActorLite的演示
  3. C#的Tag Message回顾:繁琐而危险
  4. Erlang的Actor回顾:将消息转化为逻辑执行
  5. Actor模型的本质:究竟是要解决什么问题
责任编辑:yangsai 来源: 老赵点滴
相关推荐

2009-04-01 12:00:43

ASP.NETMVC

2009-08-11 08:15:12

Silverlight

2009-10-15 09:12:29

.NET委托

2009-09-02 17:51:36

.NET委托

2009-08-18 11:08:24

.Net Framew

2024-01-15 00:30:04

Python 3语言版本

2011-05-25 09:52:40

Fedora 15

2011-06-16 15:14:17

VB.NET事件委托

2010-01-05 18:21:33

.NET Framew

2011-06-21 13:23:20

Qt 版本

2017-11-07 12:53:28

Android支持库

2010-08-03 08:52:23

委托接口

2011-05-20 16:33:47

委托接口

2012-04-13 09:13:47

Java

2015-09-28 10:12:53

Windows 10版本激活

2021-08-29 18:01:57

HTTP协议版本

2017-01-05 18:47:26

Windows 10操作系统微软

2009-06-30 08:35:51

微软Windows 7操作系统

2012-04-27 09:21:36

Ubuntu 12.0

2023-04-10 07:47:01

流程引擎Flowable
点赞
收藏

51CTO技术栈公众号