文章是很基础,但很实用,看了这篇文章,让我一下回到了2016年刚刚学委托的时候,故转之!

1.委托

委托类似于C++中的函数指针(一个指向内存位置的指针)。委托是C#中类型安全的,可以订阅一个或多个具有相同签名方法的函数指针。简单理解,委托是一种可以把函数当做参数传递的类型。很多情况下,某个函数需要动态地去调用某一类函数,这时候我们就在参数列表放一个委托当做函数的占位符。在某些场景下,使用委托来调用方法能达到减少代码量,实现某种功能的用途。

1.1.自定义委托

声明和执行一个自定义委托,大致可以通过如下步骤完成:

  1. 利用关键字delegate声明一个委托类型,它必须具有和你想要传递的方法具有相同的参数和返回值类型;
  2. 创建委托对象,并且将你想要传递的方法作为参数传递给委托对象;
  3. 通过上面创建的委托对象来实现该委托绑定方法的调用。

下面一段代码,完成了一次应用委托的演示:

    //step01:使用delegate关键字声明委托
public delegate int CalcSumDelegate(int a, int b); class Program
{
static void Main(string[] args)
{
//step03:实例化这个委托,并引用方法
CalcSumDelegate del = new CalcSumDelegate(CalcSum);
//step04:调用委托
int result = del(5, 5);
Console.WriteLine("5+5=" + result);
}
//step02:声明一个方法和委托类型对应
public static int CalcSum(int a, int b)
{
return a + b;
}
}

通过上面4个步骤,完成了委托从声明到调用的过程。接着,咱也学着大神用ILSpy反编译上面的代码生成的程序集。截图如下:

  1. 自定义委托继承关系是:System.MulticastDelegate —> System.Delegate —>System.Object。
  2. 委托类型自动生成3个方法:BeginInvoke、EndInvoke、Invoke。查资料得知,委托正是通过这3个方法在内部实现调用的。Invoke 方法,允许委托同步调用。上面调用委托的代码del(5, 5)执行时,编译器会自动调用 del.Invoke(5,5);BeginInvoke 方法,允许委托的异步调用。假如上述委托以异步的方式执行,则要显示调用dal.BeginInvoke(5,5)。

注意:BeginInvoke 和 EndInvoke 是.Net中使用异步方式调用同步方法的两个重要方法,具体用法详见微软官方示例

1.2.多播委托

一个委托可以引用多个方法,包含多个方法的委托就叫多播委托。下面通过一个示例来了解什么是多播委托:

    //step01:声明委托类型
public delegate void PrintDelegate();
public class Program
{
public static void Main(string[] args)
{
//step03:实例化委托,并绑定第1个方法
PrintDelegate del = Func1;
//绑定第2个方法
del += Func2;
//绑定第3个方法
del += Func3;
//step04:调用委托
del();
//控制台输出结果:
//调用第1个方法!
//调用第2个方法!
//调用第3个方法!
}
//step02:声明和委托对应签名的3个方法
public static void Func1()
{
Console.WriteLine("调用第1个方法!");
}
public static void Func2()
{
Console.WriteLine("调用第2个方法!");
}
public static void Func3()
{
Console.WriteLine("调用第3个方法!");
}
}

可以看出,多播委托的声明过程是和自定义委托一样的,可以理解为,多播委托就是自定义委托在实例化时通过 “+=” 符号多绑定了两个方法。

Q:为什么能给委托绑定多个方法呢?

自定义委托的基类就是多播委托MulticastDelegate ,这就要看看微软是如何对System.MulticastDelegate定义的:

MulticastDelegate拥有一个带有链接的委托列表,该列表称为调用列表,它包含一个或多个元素。在调用多路广播委托时,将按照调用列表中的委托出现的顺序来同步调用这些委托。如果在该列表的执行过程中发生错误,则会引发异常。(--摘自MSDN)

Q:为什么使用“+=”号就能实现绑定呢?

先来看上述程序集反编译后的调用委托的代码:

 

“+=”的本质是调用了Delegate.Combine方法,该方法将两个委托连接在一起,并返回合并后的委托对象。

Q:多播委托能引用多个具有返回值的方法吗?

答案是,当然能。委托的方法可以是无返回值的,也可以是有返回值的。不过,对于有返回值的方法需要我们从委托列表上手动调用。否则,就只能得到委托调用的最后一个方法的结果。下面通过两段代码验证下:

    public delegate string GetStrDelegate();
public class Program
{
public static void Main(string[] args)
{
GetStrDelegate del = Func1;
del += Func2;
del += Func3;
string result = del();
Console.WriteLine(result); //控制台输出结果:
//You called me from Func3
}
public static string Func1()
{
return "You called me from Func1!";
}
public static string Func2()
{
return "You called me from Func2!";
}
public static string Func3()
{
return "You called me from Func3!";
}
}

直接执行

正确做法:利用GetInvocationList获得委托列表上所有方法,循环依次执行委托,并处理委托返回值。

    public delegate string GetStrDelegate();
public class Program
{
public static void Main(string[] args)
{
GetStrDelegate del = Func1;
del += Func2;
del += Func3;
//获取委托链上所有方法
Delegate[] delList = del.GetInvocationList();
//遍历,分别处理每个方法的返回值
foreach (GetStrDelegate item in delList)
{
//执行当前委托
string result = item();
Console.WriteLine(result);
//控制台输出结果:
//You called me from Func1
//You called me from Func2
//You called me from Func3
}
Console.ReadKey();
}
public static string Func1()
{
return "You called me from Func1";
}
public static string Func2()
{
return "You called me from Func2";
}
public static string Func3()
{
return "You called me from Func3";
}
}

遍历执行

1.3.匿名方法

匿名方法是C#2.0版本引入的一个新特性,用来简化委托的声明。假如委托引用的方法只使用一次,那么就没有必要声明这个方法,这时用匿名方法表示即可。

    //step01:定义委托类型
public delegate string ProStrDelegate(string str);
public class Program
{
public static void Main(string[] args)
{
//step02:将匿名方法指定给委托对象
ProStrDelegate del = delegate(string str) { return str.ToUpper(); };
string result = del("KaSlFkaDhkjHe");
Console.WriteLine(result);
Console.ReadKey();
//输出:KASLFKAFHKJHE
}
}

匿名方法只是C#提供的一个语法糖,方便开发人员使用。在性能上与命名方法几乎无异。

匿名方法通常在下面情况下使用:

  1. 委托需要指定一个临时方法,该方法使用次数极少;
  2. 这个方法的代码很短,甚至可能比方法声明都短的情况下使用。

1.4.Lambda表达式

Lambda表达式是C#3.0版本引入的一个新特性,它提供了完成和匿名方法相同目标的更加简洁的格式。下面示例用Lambda表达式简化上述匿名方法的例子:

    public delegate string ProStrDelegate(string str);
public class Program
{
public static void Main(string[] args)
{
//匿名委托
//ProStrDelegate del = delegate(string str) { return str.ToUpper(); };
//简化1
//ProStrDelegate del1 = (string str) =>{ return str.ToUpper(); };
//简化2
//ProStrDelegate del2 = (str) =>{ return str.ToUpper(); };
//简化3
ProStrDelegate del3 = str => str.ToUpper();
string result = del3("KaSlFkaDhkjHe");
Console.WriteLine(result);
Console.ReadKey();
//输出:KASLFKAFHKJHE
}
}
  • 简化1:去掉delegate关键字,用"=>"符号表示参数列表和方法体之间的关系;
  • 简化2:去掉方法的参数类型;假如只有一个参数,参数列表小括号()也可省略;
  • 简化3:如果方法体中的代码块只有一行,可以去掉 return,去掉方法体的大括号{}。

1.5.内置委托

上述几种委托的使用,都没能离开定义委托类型这一步骤。微软干脆直接把定义委托这一步骤封装好,形成三个泛型类:Action<T>、Func<T>和Predicate<T>,这样就省去了定义的步骤,推荐使用。

    public class Program
{
public static void Main(string[] args)
{
//Action
Action<string> action = delegate(string str) { Console.WriteLine("你好!" + str); };
action("GG");

//Func
Func<int, int, int> func = delegate(int x, int y) { return x + y; };
Console.WriteLine("计算结果:" + func(5, 6));

//Predicate
Predicate<bool> per = delegate(bool isTrue) { return isTrue == true; };
Console.WriteLine(per(true));
}
}

它们的区别如下:

  1. Action<T>委托:允许封装的方法有多个参数,不能有返回值;
  2. Func<T>委托:允许封装的方法有多个参数,必须有返回值;
  3. Predicate<T>委托:允许封装的方法有一个参数,返回值必须为bool类型。

2.事件

委托是一种类型,事件依赖于委托,故事件可以理解为是委托的一种特殊实例。它和普通的委托实例有什么区别呢?委托可以在任意位置定义和调用,但是事件只能定义在类的内部,只允许在当前类中调用。所以说,事件是一种类型安全的委托。

2.1.定义事件

通过一个简单的场景来演示下事件的使用:

    /// <summary>
/// 音乐播放器
/// </summary>
public class MusicPlayer
{
//step01:定义 音乐播放结束 事件
public event EventHandler<EventArgs> PlayOverEvent;
public string Name { get; set; }
public MusicPlayer(string name)
{
this.Name = name;
}
//step02:定义一个触发事件的方法
public void PlaySong()
{
//模拟播放
Console.WriteLine("正在播放歌曲:" + this.Name);
for (int i = 0; i < 20; i++)
{
Console.Write(".");
Thread.Sleep(100);
}
//播放结束,则触发PlayOverEvent事件
if (PlayOverEvent != null)
{
PlayOverEvent(this, null);
}
}
}
public class Program
{
static void Main(string[] args)
{
//创建音乐播放器对象
MusicPlayer player = new MusicPlayer("自由飞翔");
//step03:注册事件
player.PlayOverEvent += player_PlayOverEvent;
//播放歌曲,结束后触发事件
player.PlaySong();
Console.ReadKey();
}
static void player_PlayOverEvent(object sender,EventArgs e)
{
MusicPlayer player = sender as MusicPlayer;
Console.WriteLine("\r\n{0}播完了!", player.Name);
}
}

程序运行结果:

总结上面事件使用的几个步骤:

  1. 用event关键字定义事件,事件必须要依赖一个委托类型;
  2. 在类内部定义触发事件的方法;
  3. 在类外部注册事件并引发事件。

public event EventHandler<EventArgs> PlayOverEvent
这句代码在MusicPlayer类定义了一个事件成员PlayOverEvent,我们说事件依赖于委托、是委托的特殊实例,所以EventHandler<EventArgs>肯定是一个委托类型。下面我们来验证一下:

EventHandler是微软封装好的事件委托,该委托没有返回值类型,两个参数:sender事件源一般指的是事件所在类的实例;TEventArgs事件参数,如果有需要创建,要显示继承System.EventArgs。

2.2.事件的本质

  MusicPlayer player = new MusicPlayer("自由飞翔");
  //注册事件
  player.PlayOverEvent += player_PlayOverEvent;
  player.PlaySong();

从上面代码我们观察到,事件要通过"+="符号来注册。我们猜想,事件是不是像多播委托一样通过Delegate.Combine方法可以绑定多个方法?还是通过反编译工具查看下。

我们看到PlayOverEvent事件内部生成了两个方法:add_ PlayOverEvent和remove_ PlayOverEvent。add方法内部调用Delegate.Combine把事件处理方法绑定到委托列表;remove方法内部调用Delegate.Remove从委托列表上移除指定方法。其实,事件本质上就是一个多播委托。

3.参考文章

[1] Edison Chou,http://www.cnblogs.com/edisonchou/p/4827578.html

[2] jackson0714,http://www.cnblogs.com/jackson0714/p/5111347.html

[3] Sam Xiao, http://www.cnblogs.com/xcj26/p/3536082.html

【转:http://www.cnblogs.com/esofar/p/5493028.html

欢迎大家关注我都我的微信 公众号,公众号涨粉丝人数,就是你们对我的喜爱程度!

【详细】【转】C#中理解委托和事件的更多相关文章

  1. 【详细】【转】C#中理解委托和事件 事件的本质其实就是委托 RabbitMQ英汉互翼(一),RabbitMQ, RabbitMQ教程, RabbitMQ入门

    [详细][转]C#中理解委托和事件   文章是很基础,但很实用,看了这篇文章,让我一下回到了2016年刚刚学委托的时候,故转之! 1.委托 委托类似于C++中的函数指针(一个指向内存位置的指针).委托 ...

  2. C#知识体系(二)用案例来理解委托与事件

    上一篇博客讲到了LinQ和lambda的常用方法 还有很多我们未知但c#设计团队已经为我们封装好的类和方法.随着我们不断的熟悉C#语言,渐渐的就会接触到其他的知识点,委托.事件.反射.线程.同步,异步 ...

  3. C#学习之初步理解委托、事件、匿名方法和Lambda

    最经在学习LinqtoSql,然后扯到Lambda表达式,然后扯到匿名方法,然后扯到委托,最后扯到事件处理...后来发现对委托这个概念和事件处理这个过程理解得不是很清晰,遂得一下学习笔记.那里说得不对 ...

  4. C#基础篇 - 理解委托和事件

    1.委托 委托类似于C++中的函数指针(一个指向内存位置的指针).委托是C#中类型安全的,可以订阅一个或多个具有相同签名方法的函数指针.简单理解,委托是一种可以把函数当做参数传递的类型.很多情况下,某 ...

  5. 【转载】详细解读C#中的 .NET 弱事件模式

    你可能知道,事件处理是内存泄漏的一个常见来源,它由不再使用的对象存留产生,你也许认为它们应该已经被回收了,但不是,并有充分的理由. 在这个短文中(期望如此),我会在 .Net 框架的上下文事件处理中展 ...

  6. Unity3D中使用委托和事件

    前言: 本来早就想写写和代码设计相关的东西了,以前做2DX的时候就有过写写观察者设计模式的想法,但是实践不多.现在转到U3D的怀抱中,倒是接触了不少委托事件的写法,那干脆就在此总结一下吧. 1.C#中 ...

  7. C#中委托和事件

    目 录 将方法作为方法的参数 将方法绑定到委托 更好的封装性 限制类型能力 范例说明 Observer 设计模式简介 实现范例的Observer 设计模式 .NET 框架中的委托与事件 为什么委托定义 ...

  8. C# 关于委托和事件的妙文:通过一个例子详细介绍委托和事件的作用;Observer模式简介

    委托和事件在 .Net Framework中的应用非常广泛,然而,较好地理解委托和事件对很多接触C#时间不长的人来说并不容易.它们就像是一道槛儿,过了这个槛的人,觉得真是太容易了,而没有过去的人每次见 ...

  9. c#关于委托和事件(二)(介绍的很详细)

    using System;using System.Collections.Generic;using System.Text; namespace Delegate {    // 热水器    p ...

随机推荐

  1. XyTalk企业即时通讯IM开始开源

    网址: https://gitee.com/475660/xyTalk-pc https://github.com/xy-Group/xyTalk-pc Xy.Platform是一个高性能.可扩展的企 ...

  2. Mac下搭建react及bable

    1.安装node 下载: https://nodejs.org/en/download/ 测试--->返回版本号即为安装成功: $ node -v $ npm -v 2.安装全局create-r ...

  3. 2019年19道java经典面试题(附答案)

    1.不可变对象 指对象一旦被创建状态不能再改变.任何修改都会创建一个新的对象,如 String.Integer及其它包装类. 2.能否创建一个包含可变对象的不可变对象? 可以.不要共享可变对象的引用就 ...

  4. OSI七层模型和tcp/ip四层模型对比

    OSI 与TCP/IP 模型对比 OSI 协议层名称 TCP/IP 协议层名称 封装的单元 功能描述 TCP/IP协议 应用层(Application) 应用层(Application) 数据 应用程 ...

  5. Linux_CentOS-服务器搭建 <三> 补充

    今天 才发现,服务器上 JDK 都没有好好的安装下.在这里补充说下. 1.看看机子上JDK的安装了多少 $ rpm -qa |grep java 会出现类似: java-1.6.0-openjdk-1 ...

  6. 深入理解L1、L2正则化

    过节福利,我们来深入理解下L1与L2正则化. 1 正则化的概念 正则化(Regularization) 是机器学习中对原始损失函数引入额外信息,以便防止过拟合和提高模型泛化性能的一类方法的统称.也就是 ...

  7. k8s全栈监控之metrics-server和prometheus

    一.概述 使用metric-server收集数据给k8s集群内使用,如kubectl,hpa,scheduler等 使用prometheus-operator部署prometheus,存储监控数据 使 ...

  8. ThreadLocalMap里Entry为何声明为WeakReference?

    Java里,每个线程都有自己的ThreadLocalMap,里边存着自己私有的对象.Map的Entry里,key为ThreadLocal对象,value即为私有对象T.在spring MVC中,常用T ...

  9. JavaWeb学习(二十二)———EL表达式

    一.EL表达式简介 EL 全名为Expression Language.EL主要作用: 1.获取数据 EL表达式主要用于替换JSP页面中的脚本表达式,以从各种类型的web域 中检索java对象.获取数 ...

  10. Algolia使用教程 , 超详细傻子看都会

    框架描述 发现网上Algolia这块的资料较少,就花了点时间从官网上整理了下,总结了几项常用的功能用法. 现在比较有名的Algolia提供了云搜索的服务.具体办法是我们将数据库的信息以JSON的格式上 ...