• 艾伟_转载:解惑答疑:C#委托和事件 狼人:


      相信很多人一直为C#委托和事件所困惑,尤其是C#的初学者,学到这一部分会感觉比较困难,很有可能就放弃了,而且.NET对委托和事件封装得挺好,一般都不怎么用到自定义的委托和事件,所以放弃学习该技术就有了借口!

      网上也有不少此类的文章,最具代表性的是张子阳的C#中的委托和事件这篇,写得的确很好,得到很多读者的赞赏,但我 看评论,还是发现了些问题,因为有不少读者是看了一遍又一遍,每次感觉都蛮好,可是隔一段时间,对“委托和事件”又迷糊了,于是又来看!我真搞不懂,为什 么会出现这种情况!后来想想,文章虽好,但总结的地方没有把重点列出来;再者,读者跟着作者的思路,的确能把文章看懂,但是读者自己不得要领,没有真正弄 明白,因此就出现了上面提到的状况!

      C#委托和事件真的难吗,其实不然。要搞懂它,第一,要看你的理解能力;第二,要看你怎么理解它。如果你觉得理解起来比较困难,那我们可以换种理解方式,也许能很好地理解它了!其实委托和事件的确不难,大牛级别的甚至都不屑写此类文章!

      为什么会有委托?

      委托其实就是个方法指针,拥有同样参数和返回值的任何方法都能传给委托;委托能够消除条件分支语句,不需要根据if、case这些语句来判断具体调用哪个方法!而委托又是从观察者模式演化而来,这里推荐阅读TerryLee的这篇观察者模式文章。  

      前面说的的就算不理解也不要紧,关键是理解方法、委托、事件之间的关系。可以这样说,方法是“委托”给委托的,而委托是“委托”给事件的。可以将事件看成是委托的一个容器,里面可以加一连串的委托!这样来理解,那所有的事情就都解决了!

      当然,我们都是在某个方法中触发事件,事件将其交给委托,委托再交给方法,方法再进行实际的操作,与上面的步骤刚好相反!其实触发事件的目的就是触发具体方法!

      再来说说委托的好处(上面没举例子),比如你开发了一个电子商务平台,后台有管理商品的功能,而商品信息有七八列或者更多,包括编号、商品名称、价 格、上架时间等等,该信息又能根据任意一列来进行排序!如果没有委托,我们将根据点击某列所产生的信息,将这信息传给某个排序方法,而这个排序方法会接受 传来的信息作为参数,再根据内部的分支语句if、case等来判断具体采用哪个排序方法,这样的话,逻辑变复杂,这过程当中还要做很多无用功(因为很有可 能要进行多次判断才能找到要真正执行的方法),而且如果我们将来再增加列,又得增加分支语句,违背了“开放—封闭”原则,维护起来比较麻烦!有了委托,我 们不需要传递任何参数,直接将具体方法传给委托即可,增加列则只要增加一个新方法,爽!我们完全可以通过委托来调用方法,那为什么还要事件呢?事件其实是 对委托进行一种限制,使其无法使用“=”赋值运算符(如果使用则在编译时产生错误),只能使用“+=”或者“-=”运算符,这就防止了程序员误将原先的委 托链给覆盖掉,另外delegate类从MulticastDelegate(多路广播委托)继承而来,所以可以将多个委托赋给同一个事件!

      最后,列一串代码把上面的概念理清一下

    class Program
    {
    static void Main(string[] args)
    {
     XiaoBai xiaobai
    = new XiaoBai();
    //Google公司
    ItCompany google = new ItCompany("谷歌中国", "CTO", xiaobai);
    //微软公司
    ItCompany microsoft = new ItCompany("微软中国", "架构师", xiaobai);
    //花旗银行
    FinanceCompany AmericaBank = new FinanceCompany("花旗银行", "金融分析师", xiaobai);
    //委托的好处,可以应用于不同的类的不同方法
    //方法“委托”给委托,委托“委托”给事件
    //委托类型与事件声明时的委托类型相同
    //因为是传引用,所以方法后面不能带括号,带括号则是调用方法了
    //一个委托可以搭载多个方法,一个事件则拥有一个委托链
    xiaobai.Update += new TheEventHandler(google.ComeToItCompany);
    xiaobai.Update
    += new TheEventHandler(microsoft.ComeToItCompany);
    xiaobai.Update
    += new TheEventHandler(AmericaBank.ComeToFinanceCompany);
    xiaobai.SubjectState
    = "我小白过来应聘职位啦!";
    //发出通知,触发事件
    xiaobai.Notify();
    //以下代码与上面相似
    XiaoHua xiaohua = new XiaoHua();
    ItCompany microsoft2
    = new ItCompany("微软总公司", "CEO", xiaohua);
    FinanceCompany ChinaBank
    = new FinanceCompany("中国央行", "财务部总经理", xiaohua);
    xiaohua.Update
    += new TheEventHandler(microsoft2.ComeToItCompany);
    xiaohua.Update
    += new TheEventHandler(ChinaBank.ComeToFinanceCompany);
    xiaohua.SubjectState
    = "我小华过来应聘职位啦!";
    xiaohua.Notify()
    Console.ReadLine();
    }
    }
    //通知者接口
    interface Subject
    {
    void Notify();

    string SubjectState
    {
    get;
    set;
    }
    }
     
    //事件处理程序的委托,相当于一个类(在编译成IL后确确实实是类)或者方法指针,
    与常规类定义不同,带参数和返回值

    delegate void TheEventHandler();
    //小白
    class XiaoBai : Subject
    {
    //声明一事件Update,类型为委托TheEventHandler
    public event TheEventHandler Update;
    private string action; //用Notify方法触发事件
    public void Notify()
    {
    Update();
    }
    public string SubjectState
    {
    get { return action; }
    set { action = value; }
    }
    }
    //小华
    class XiaoHua : Subject
    {
    //声明一事件Update,类型为委托TheEventHandler
    public event TheEventHandler Update;
    private string action;
    //用Notify方法触发事件
    public void Notify()
    {
    Update();
    }
    public string SubjectState
    {
    get { return action; }
    set { action = value; }
    }
    }
    //IT行业
    class ItCompany
    {
    private string companyname;
    private string job;
    private Subject sub;

    public ItCompany(string _companyname, string _job, Subject _sub)
    {
    companyname
    = _companyname;
    job
    = _job;
    sub
    = _sub;
    }
    //参数和返回值与委托TheEventHandler一致
    public void ComeToItCompany()
    {
    Console.WriteLine(
    "{0} {1}: 来我们公司做{2}!", sub.SubjectState,
    companyname, job);
    }
    }
    //金融行业
    class FinanceCompany
    {
    private string companyname;
    private string job;
    private Subject sub;

    public FinanceCompany(string _companyname, string _job, Subject _sub)
    {
    companyname
    = _companyname;
    job
    = _job;
    sub
    = _sub;
    }
    //参数和返回值与委托TheEventHandler一致
    public void ComeToFinanceCompany()
    {
    Console.WriteLine(
    "{0} {1}: 来我们公司做{2}!", sub.SubjectState,
    companyname, job);
    }
    }

      通过以上总结,我相信大家对C#委托和事件应该可以更好地理解了!当然,委托的知识不止这些,还会用到检查空值、异常处理和多线程处理等等,这篇文章仅在 解惑(我也不高兴浪费太多的时间来具体讲解)!如果你想更好地掌握委托和事件,可以看下上面提到的张子阳的两篇文章或者买本《C#本质论》仔细研读;如果 你想了解观察者模式,可以看下上面提到的TerryLee那篇文章;如果你还没有学习设计模式或者刚刚开始学习,我建议阅读《大话设计模式》;如果你学习 设计模式有一段时间了,我建议阅读《设计模式:基于C#的工程化实现及扩展》!祝各位程序员好运!

  • 相关阅读:
    浅析@Deprecated
    微信小程序开发系列一:微信小程序的申请和开发环境的搭建
    HTTP 200 OK和HTTP 304 Not modified的由来
    深入理解Java的整型类型:如何实现2+2=5?
    聊聊JavaScript和Scala的表达式 Expression
    Java异常处理:如何写出“正确”但被编译器认为有语法错误的程序
    如何在ABAP里用函数式编程思想打印出非波拉契Fibonacci(数列)
    使用JavaScript ES6的新特性计算Fibonacci(非波拉契数列)
    JavaScript面试系列:JavaScript设计模式之桥接模式和懒加载
    Cordova应用的JavaScript代码和自定义插件代码的调试
  • 原文地址:https://www.cnblogs.com/waw/p/2157100.html
Copyright © 2020-2023  润新知