深刻理解:C#中的委托、事件

C#中的事件还真是有点绕啊,以前用JavaScript的我,理解起来还真是废了好大劲!刚开始还真有点想不明白为什么这么绕,想想和JS的区别,最后终于恍然大悟!

C#中事件绕的根本原因:

  1. C#的方法,它不是一个类型,它只是其它类型的成员;
  2. C#是一个强类型的语言,定义方法时,它的参数必须指定类型,如 public void add(int n){...};

所以,一个方法不能直接作为其它方法的参数,把一个方法名作为参数,无法指定类型啊,会报错!那我就想啊,既然不能直接传入,那我传入整个对象总可以吧,通过传进来的对象来执行该方法,如下代码:

using System;
namespace MyEventTest
{
    public class SomeClass
    {
        public void Start(int a) { Console.WriteLine("Go:{0}",a); }
    }
    public class Publisher
    {
        public void StartEvent(int a, SomeClass sc)
        {
            if (sc != null)
            {
                sc.Start(a);  //触发回调方法
            }
        }
    }
    public class MainClass
    {
        static void Main()
        {
            SomeClass some = new SomeClass();
            Publisher p = new Publisher();
            p.StartEvent(5,some); //Go:5
        }
    }
}

以上方法确实可以,但C#不完全是这样实现事件的,因为方法的特殊性,C#引入了委托的概念,让委托对象来代表方法作为其它方法的参数;而事件对象,其实就是一个委托对象。下面先介绍一下委托:

委托

对应于以上方法:

public void Start(int a) { Console.WriteLine("Go:{0}",a); }

我们可以定义一个委托类型:

public delegate void MyDel(int a);

委托的定义:

  1. 委托MyDel它是一个类型,类型名就是MyDel;定义委托相当于定义一个新类,委托在后台实现为派生自System.Delegate类。
  2. 定义委托,就是告诉编译器该委托将表示哪种方法(返回值类型+方法签名),该方法可以是任意类型的实例方法、静态方法,只要方法的签名、返回值类型与委托匹配,那么该委托的实例就可以引用这些方法。
  3. 使用委托,必须创建该委托的实例,并为它指定要引用的方法,如: MyDel d = some.Start;注意这里不是 some.Start() ;
  4. 委托对象支持"+","+="来为它添加更多的方法引用,而"-","-="则是删除引用;
  5. 引用了多个方法的委托就叫多播委托,多播委托派生自基类 System.MulticastDelegate 类,它是 System.Delegate 类的子类
  6. 注意:只要委托对象还存在对方法的引用,它就一直占用内存哦!我想可以用 d = null; 来释放委托对象d;
  7. 可以对委托对象执行调用,如: d(5); 它将把调用传递给它所引用的方法 some.Start(5); ,对于多播委托,它将按顺序调用它引用的所有方法,但如果其中一个方法抛出异常,且没在方法内部处理,则将会将异常往外抛出,之后的方法调用将终止。

    使用委托的规则:

  8. 委托是和类一个级别的,可以在能定义类的任何地方定义委托;
  9. 委托不支持继承;
  10. 可以为委托类型定义任意常见的访问修饰符;
  11. 委托对象所引用的方法也可以是匿名方法、Lambda 表达式;
  12. 多播委托的返回值类型必须是void,否则就只能得到委托调用的最后一个方法的结果。
  13. 在.NET 4.0中,委托开始支持协变与逆变,这样一来,定义委托类型时的签名可以和所要引用的方法的签名不完全匹配(不同类型之间必须是派生关系)
  14. 委托支持泛型,.NET预定义了两个泛型版本的委托:
    • Action< T > 委托表示引用一个返回值类型为void的方法,根据参数个数存在不同的变体版本;如: Action<in T1, in T2>
    • Func< T > 委托表示引用一个带返回值类型的方法,根据参数个数存在不同的变体版本;如: Func<in T1, out TResult> 1个参数T1和返回值类型TResult。

事件

说完了委托的概念,就可以继续讲事件了,因为事件是基于委托的!

事件的概念:

  • 类或对象可以通过事件向其他类或对象通知发生的相关事情。
  • 发送事件的类称为“发行者”,接收事件的类称为“订阅者”。(就是设计模式中的订阅发布者模式);
  • 一个事件可以有多个订阅者。 一个订阅者可处理来自多个发行者的多个事件。如果一个事件有多个订阅者,当引发该事件时,会同步调用多个事件处理程序。也可异步调用。

.NET Framework 类库中的所有事件均基于 EventHandler 委托,还有泛型版本 EventHandler<EventArgs> ,这个委托是.NET预定义的,不需要我们定义,可以直接用它来实例化一个事件对象,定义如下:

参数object sender对象是对发布者的实例的引用,EventArgs e对象主要用来存储事件数据

public delegate void EventHandler(object sender, EventArgs e); //EventArgs主要用来存储事件数据

public delegate void EventHandler&lt;TEventArgs&gt;(object sender, EventArgs e);

虽然在自定义的类中的事件可基于任何有效委托类型,但是,通常建议使用.NET预定义事件委托类型让事件基于 .NET 标准事件模式

下面是我总结的发布基于 .NET 标准事件模式的4个步骤:

第1步:在发布者类中实例化委托事件,并定义一个实例方法,用来调用委托事件(因为委托事件只能通过定义它的类的实例来调用)。

定义发布者类之前可先定义一个用来存储事件数据的类(它必须派生于 EventArgs 基类),如下:

注意:在方法StartEvent()中,声明了一个变量,来保存事件对象的副本,这样在取得事件对象的副本后,到触发事件时,这段时间内,这个事件副本就不会受其它线程的影响。如:在此期间,其它线程注销了回调方法,那么MyEvent就为null了,这时再触发事件将引发错误。(这就是线程安全的事件,当然还可以通过锁机制,或者为事件对象始终引用一个空方法)

public class MyEventArgs: EventArgs  //定义存储事件数据的类
{
    public int Current{get;set;}
}
public class Publisher
{
    public event EventHandler<MyEventArgs> MyEvent; //第1步:实例化委托事件
    public int Sum{get;set;}
    public void StartEvent(int a)
    {
          var EventCopy = MyEvent; //每次都取一个副本
          MyEventArgs args = new MyEventArgs();
          args.Current = a;
          this.Sum += a;
          if (EventCopy != null)
          {
               EventCopy(this,args);  //调用事件
          }
    }
}

第2步:定义订阅者类,在该类中定义和委托事件相匹配的方法(事件触发时,实际要执行的方法)

public class Subscriber
{
     public void Dosomething1(object obj, MyEventArgs e)
     {
            Publisher p = (Publisher)obj;
            Console.WriteLine("Meg: Sum = {0}, Current = {1}", p.sum, e.Current);
     }
     public void Dosomething2(object obj, MyEventArgs e)
     {
     }
}

第3步:在客户端代码中,在发布者类的实例上为委托事件注册回调方法

public class MainClass
{
    static void Main()
    {
         Publisher p = new Publisher{ Sum = 0 };
         Subscriber sub = new Subscriber();
         p.MyEvent += sub. Dosomething1;  //注册回调方法
         p.MyEvent += sub. Dosomething2;

         p. StartEvent( 5 ); //调用方法,间接触发事件

         p.MyEvent -= sub. Dosomething1;  //取消注册
    }
}

要点:事件对象其实就是一个委托对象,把事件当委托来看,就比较容易理解了!不要被Event这个单词给蒙蔽了!

介绍完了!下回将介绍C#中的其它一些较难理解的概念!

时间: 2024-12-20 17:18:16

深刻理解:C#中的委托、事件的相关文章

深刻理解Java中形参与实参,引用与对象的关系

声明:本博客为原创博客,未经允许,不得转载!原文链接为http://blog.csdn.net/bettarwang/article/details/30989755 我们都知道,在Java中,除了基本数据类型之外,其他的都是引用类型,当它们作为函数参数时,传递的也是引用,通过引用可以改变对象的值,很多人便因此而忽略形参与实参,引用与对象的关系问题.废话不多说,先看下面一个例子: import java.util.*; public class Student { private String

深刻理解Java中final的作用(一):从final的作用剖析String被设计成不可变类的深层原因

声明:本博客为原创博客,未经同意,不得转载!小伙伴们假设是在别的地方看到的话,建议还是来csdn上看吧(原文链接为http://blog.csdn.net/bettarwang/article/details/26744661),看代码和提问.讨论都更方便. Java中final的作用主要表如今三方面:修饰变量.修饰方法和修饰类.以下就从这两个方面来解说final的作用.在文末从final及类的设计安全性出发,论述了Java中String为何要被设计成不可变类. 1.final修饰变量 fina

深刻理解Python中的元类

译注:这是一篇在Stack overflow上很热的帖子.提问者自称已经掌握了有关Python OOP编程中的各种概念,但始终觉得元类(metaclass)难以理解.他知道这肯定和自省有关,但仍然觉得不太明白,希望大家可以给出一些实际的例子和代码片段以帮助理解,以及在什么情况下需要进行元编程.于是e-satis同学给出了神一般的回复,该回复获得了985点的赞同点数,更有人评论说这段回复应该加入到Python的官方文档中去.而e-satis同学本人在Stack Overflow中的声望积分也高达6

深刻理解Python中的元类(metaclass)以及元类实现单例模式

深刻理解Python中的元类(metaclass)以及元类实现单例模式 在看一些框架源代码的过程中碰到很多元类的实例,看起来很吃力很晦涩:在看python cookbook中关于元类创建单例模式的那一节有些疑惑.因此花了几天时间研究下元类这个概念.通过学习元类,我对python的面向对象有了更加深入的了解.这里将一篇写的非常好的文章基本照搬过来吧,这是一篇在Stack overflow上很热的帖子,我看http://blog.jobbole.com/21351/这篇博客对其进行了翻译. 一.理解

深刻理解C#中资源释放

今天我的一个朋友看到我写的那篇<C#中用AJAX验证用户登录>时,给我指出了点小毛 病.就是在用户登录时,如果用户登录失败,在下面这段代码中,都会new出来一个User对象,如果连续登录失败多次,就会生成多个User对象,而它们 在登录失败后已经无用了,依然占据着内存,就算是C#有垃圾回收机制,但不确定什么时候对这些对象进行回收.然后去网上找了一篇C#资源释放的文章,讲的很透彻,和大家分享一下. using System;using System.Collections.Generic;usi

深刻理解Java中的String、StringBuffer和StringBuilder的区别

首先简单地来梳理一下Java中String.StringBuffer和StringBuilder各自的含义. 1.String类 首先,它是线程安全的,即可以用于多线程编程中: 其次,String类的对象是不可变的,即在定义时就确定了,类似String str="Hello";str+="Java";的语句其实是生成了新的对象,只是我们未察觉到而已.但是注意在大量的字符串新建对象时消耗就很可观,这时必须考虑采用StringBuffer或StringBuilder,否

深刻理解Java中的String、StringBuffer和StringBuilder的差别

声明:本博客为原创博客,未经同意.不得转载!小伙伴们假设是在别的地方看到的话,建议还是来csdn上看吧(链接为http://blog.csdn.net/bettarwang/article/details/26412497),看代码和提问.讨论都更方便. 首先简单地来梳理一下Java中String.StringBuffer和StringBuilder各自的含义. 1.String类 首先.它是线程安全的,即能够用于多线程编程中. 其次,String类的对象是不可变的,即在定义时就确定了,类似St

【Swing】理解Swing中的事件与线程

talk is cheap , show me the code. Swing中的事件 事件驱动 所有的GUI程序都是事件驱动的.Swing当然也是. GUI程序不同于Command Line程序,一个很大的区别是程序执行的驱动条件:命令行程序是接受用户输入的文本参数,对命令解析,然后通过类似switch的选择来执行不同的功能模块.而GUI程 序就不一样了.GUI程序由界面元素组成,如Button,CheckBox,TextArea,等等.用户操作不同的组件,就会引发不同的事件,然后, 程序编写

理解Javascript中的事件绑定与事件委托

最近在深入实践js中,遇到了一些问题,比如我需要为动态创建的DOM元素绑定事件,那么普通的事件绑定就不行了,于是通过上网查资料了解到事件委托,因此想总结一下js中的事件绑定与事件委托. 事件绑定   最直接的事件绑定:HTML事件处理程序 如下示例代码,通过节点属性显式声明,直接在HTML中,显式地为按钮绑定了click事件,当该按钮有用户点击行为时,便会触发myClickFunc方法. /* html */ <button id="btn" onclick="myCl

一个demo让你彻底理解Android中触摸事件的分发

注:本文涉及的demo的地址:https://github.com/absfree/TouchDispatch 1. 触摸动作及事件序列 (1)触摸事件的动作 触摸动作一共有三种:ACTION_DOWN.ACTION_MOVE.ACTION_UP.当用户手指接触屏幕时,便产生一个动作为ACTION_DOWN的触摸事件,此时若用户的手指立即离开屏幕,会产生一个动作为ACTION_UP的触摸事件:若用户手指接触屏幕后继续滑动,当滑动距离超过了系统中预定义的距离常数,则产生一个动作为ACTION_MO