声明:这篇博客翻译自:https://www.codeproject.com/Articles/1061085/Delegates-Multicast-delegates-and-Events-in-Csharp

第一次翻译英文博客,由于水平(技术水平+英语理解能力)有限/不足,肯定会有所疏漏/错误,请及时指正。

介绍:

在网络上搜索一下关于C#代理,事件,多播代理的文章,很多很多。不过还是有些地方讲的不明白/透彻。这篇博客将以简单易懂的方式来讲解这3个概念。下面先回答一下:

  • 什么是delegate,在哪里使用delegate?
  • 什么是multicast delegate?
  • 何时使用delegate,Event

简要的答案:

下面的图解解释了delegate,multicase delegate, event之间的联系。

  • Delegate是方法的指针,可以用作回调函数;
  • Multicast delegate用作调用多个回调函数;
  • Event封装了delegate,并且实现了订阅-发布模型;
  • Event和multicase delegate都是delegate,因此delegate是event和multicase delegate的基础;

在本篇文章的剩余部分,我们将来更详细的理解上面这些话。

Delegate是方法指针

“Delegate是方法的指针”,你可以通过delegate来调用指向的方法。

通过下面3步创建delegate:

声明delegate,PS: 在这一步你将使用delegate关键字来声明委托,注意必须保证delegate的签名和需要指向的函数/方法签名一样/一致。例如下面代码中“SomeMethod()"无返回值且无输入参数。因此示例代码中的"SomeMethodPtr()"delegate定义是合适的;

创建一个delegate对象,PS: 当你创建了一个delegate之后,需要创建一个delegate的对象才能使用它,参考代码中注释Step2;

调用delegate,PS: 通过调用delegate的Invoke方法调用"SomeMethod".

    delegate void SomeMethodPtr();  // 1. Declare delegate

    static void Main(string[] args)
{
SomeMethodPtr ptrObject = SomeMethod; // 2. Create object of delegate
ptrObject.Invoke(); // 3. Invoke the delegate
} static void SomeMethod()
{
// some code
}

为什么要使用这种见解调用呢?

目前我们理解了delegate是方法/函数的指针。通过delegate间接调用一个方法有什么好处呢?

当我们需要调用的code在另外一个程序集中,例如下面的文件搜索类在一个其他assebmly中。PS:这个文件搜索代码只是一个模拟代码。

public class SearchFile
{
public void Search()
{
// File search is happening
for(int i=;i<;i++)
{
string str = "File " + i;
}
}
}

现在我们在另外一个单独的Console应用中调用上面的代码,当搜索完毕后,立刻通知UI并显示搜索的文件名。

static void Main(string[] args)
{
SearchFile fl = new SearchFile();
fl.Search();
}

换句话所我们需要一个CALLBACK(回调函数)当File Search结束后通知Console程序。这就是为什么说delegate是用来做回调函数的。

很多开发者想为什么不在Searh()方法中直接调用“Console.Write”进行输出呢。这样做会将UI技术和核心代码有耦合。如果我们把这段代码给WPF/WinForms程序调用,Console.Write做UI输出不适用.

    public class SearchFile
{
public void Search()
{
// File search is happening
for (int i = ; i < ; i++)
{
string str = "File" + i;
Console.Write(str);
}
}
}

使用delegate实现一个回调函数
在SearchFile类中实现delegate,第一件事是暴露一个delegate的调用。在下面的SearchFile类中,定义一个WheretoCall的delegate,仔细观察Step1,2中的代码,此时的WheretoCall是空的,任何一个客户端程序想要有一个回调事件,传递一个签名一致的方法即可。

    public class SearchFile
{
public delegate void WheretoCall(string status); // Step1 public WheretoCall wheretoCall = null; // Step 2 public void Search()
{
// File search is happening
for (int i = ; i < ; i++)
{
string str = "File" + i;
wheretoCall(str); // Step 3
}
}
}

现在调用上述代码时只需要传递一个方法的引用到delegate指针即可。请看下面代码中Step 1

    static void Main(string[] args)
{
SearchFile fl = new SearchFile();
fl.wheretoCall = CallHere; // Step 1
fl.Search();
} static void CallHere(string message)
{
Console.Write(message);
}

Multicast delegate(多播委托)

上面图示中SearchFile"类发送通知(广播)到3个订阅方法。我们可以把这种模式称为订阅发布架构。SearchFile类是发布者。

    static void CallHereToWriteToFile(string message)
{
System.IO.File.WriteAllText(@"c:\somtext.txt", message);
}
static void CallHereForConsole(string message)
{
Console.WriteLine(message);
}
static void CallHereToWriteInternally(string message)
{
messages.Add(message);
}

为了实现上述目标,我们不需要修改SearchFile类,在客户端调用时使用“+=”将方法分配给“wheretoCall”代理。如果你不想订阅,使用"-="取消订阅即可。

    static void Main(string[] args)
{
SearchFile fl = new SearchFile();
fl.wheretoCall += CallHereForConsole;
fl.wheretoCall += CallHereToWriteToFile;
fl.wheretoCall += CallHereToWriteInternally;
fl.Search();
}

Event -- 封装后的delegate

通过multicast delegate实现的订阅-发布者模式有一个严重的问题,订阅者可以修改这个delegate。在一个真实的订阅-发布者模式/广播模式中,订阅者只能订阅/取消订阅。

下面的代码中,订阅者可以将发布者delegete设置为NULL,并且可以自由调用delegate。

    static void Main(string[] args)
{
SearchFile f1 = new SearchFile();
f1.wheretoCall += CallHereForConsole;
f1.wheretoCall.Invoke("status"); // The client can invoke.
f1.wheretoCall = null; // The delegate can be modified.
f1.wheretoCall += CallHereToWriteInternally;
f1.Search();
}

下面使用event关键字封装delegate,

    public class SearchFile
{
public delegate void WheretoCall(string status); // Step1 public event WheretoCall wheretoCall = null; // Step 2 public void Search()
{
// File search is happening
for (int i = ; i < ; i++)
{
string str = "File" + i;
wheretoCall(str); // Step 3
}
}
}

编译客户端代码,出现如下错误:

这个错误的意思是对于一个Event事件你只能订阅(+=)或者取消订阅(-=)。

总结:

[C#] Delegate, Multicase delegate, Event的更多相关文章

  1. C#中的委托(Delegate)和事件(Event)

    原文地址:C#中的委托(Delegate)和事件(Event) 作者:jiyuan51 把C#中的委托(Delegate)和事件(Event)放到现在讲是有目的的:给下次写的设计模式--观察者(Obs ...

  2. (转)C#中的委托(Delegate)和事件(Event)

    转自:http://blog.chinaunix.net/uid-576762-id-2733751.html   把C#中的委托(Delegate)和事件(Event)放到现在讲是有目的的:给下次写 ...

  3. C# delegate和C# event

    在基于Windows平台的程序设计中,事件(event)是一个很重要的概念.因为在几乎所有的Windows应用程序中,都会涉及大量的异步调用,比如响应点击按钮.处理Windows系统消息等,这些异步调 ...

  4. 重温委托(delegate)和事件(event)

    1.delegate是什么 某种意义上来讲,你可以把delegate理解成C语言中的函数指针,它允许你传递一个类A的方法m给另一个类B的对象,使得类B的对象能够调用这个方法m,说白了就是可以把方法当作 ...

  5. 关于C# 委托(delegate)与事件(event)的用法及事例

    C#中的委托和事件对于新手可能会有一点难理解,所以先从一个小例子入手,以便能更好的理解其如何使用.有一个学生每天定闹钟在早上6点起床,所以当每天早上6点的时候,闹钟就会响起来,从而学生才会按时起床. ...

  6. C# Delegate 匿名 Delegate

    C#6.0新添加了 lambda的强力支持,用lambda的确可以节省好多代码,让代码看起来更简洁,更直观: 这里做一个笔记,C#的匿名委托 Demo class Program { static v ...

  7. c# 关键字delegate、event(委托与事件)[MSDN原文摘录][1]

    A delegate is a type that safely encapsulates a method, similar to a function pointer in C and C++. ...

  8. .NET 中易混淆的概念(Delegate vs Event)

    事件(event)是一个非常重要的概念,我们的程序时刻都在触发和接收着各种事件:鼠标点击事件,键盘事件,以及处理操作系统的各种事件.所谓事件就是 由某个对象发出的消息.比如用户按下了某个按钮,某个文件 ...

  9. delegate vs event

    What are the differences between delegate and an event? An event declaration adds a layer of abstrac ...

随机推荐

  1. HDU2888 Check Corners(二维RMQ)

    有一个矩阵,每次查询一个子矩阵,判断这个子矩阵的最大值是不是在这个子矩阵的四个角上 裸的二维RMQ #pragma comment(linker, "/STACK:1677721600&qu ...

  2. HDU 4123 Bob’s Race(RMQ)

    题意是说给出一棵树,N(10^5)个顶点,以及每条边的权值,现在需要选择连续的K个点(顶点编号连续),可以被选出来的条件是: 若d[i]代表顶点i到树上其他点的距离的最大值,使得区间[a, b]的d值 ...

  3. ambassador 学习四 grpc 处理

    实际上都是envoy 的功劳 基本环境安装参考相关文档即可 参考demo proto code syntax = "proto3"; option java_multiple_fi ...

  4. Django ORM介绍 和字段及字段参数

    Object Relational Mapping(ORM) ORM介绍 ORM概念 对象关系映射(Object Relational Mapping,简称ORM)模式是一种为了解决面向对象与关系数据 ...

  5. 【转】ubuntu中没有/etc/inittab文件探究

    原文网址:http://blog.csdn.net/gavinr/article/details/6584582 linux 启动时第一个进程是/sbin/init,其主要功能就是软件执行环境,包括系 ...

  6. ASP.NET MVC基础入门.

    一:ASP.NET MVC 简介 1:asp.net mvc 是一种构建web应用程序的框架,他将一般的MVC(Model--View--Controller)模式应用于asp.net框架. 2:as ...

  7. vs2005+WinCE模拟器+ActiveSync调试WinCE程序

    来源:http://www.cnblogs.com/xjimmyshcn/archive/2011/07/19/2111087.html 一.WinCE 模拟器通过ActiveSync 6.1(即Wi ...

  8. 黄聪:WordPress 多站点建站教程(二):后台(管理网络)设置详解,如何管理子站的用户、主题、插件、设置等功能

    建立好了子站,我们需要有个地方配置所有子站的主题.插件等功能,我们可以在后台看到 我的站点--管理网络 如下图: 在 管理网络--仪表盘 里面,我们可以创新用户和站点,也提供了查询功能. 要注意的是: ...

  9. 关于filter web api mvc 权限验证 这里说的够详细了。。。

    参考:http://www.cnblogs.com/willick/p/3331520.html Filter(筛选器)是基于AOP(面向方面编程)的设计,它的作用是对MVC框架处理客户端请求注入额外 ...

  10. 正规式->最小化DFA说明

      整体的步骤是三步: 一,先把正规式转换为NFA(非确定有穷自动机), 二,在把NFA通过"子集构造法"转化为DFA, 三,在把DFA通过"分割法"进行最小化 ...