去年自学C#用的教程是入门级的《学通C#的24堂课》,教材里面也没有提到委托和事件,工作中也没怎么用到。后来一次在网上看了一些大牛的博客,读完之后感觉懵懵懂懂,似懂非懂,过了两三天之后,却又全然忘记了。毕竟学习这事,温故而知新,学了不用,自然忘得也很快。对于如我一样的初学者来说,较好地理解委托和事件并是一件容易的事。其实掌握了的人,会觉得也没什么,而没有掌握的人,每次见到委托和事件就会觉得很畏惧。前段时间看到张旭亮老师的博客中关于.NET 开发系列PPT中提到一个观点,没学会委托就等不会.NET。我深受激励,所以这次下定决心认认真真学习了一下,并将自己的一些理解记录下来。

   委托,字面上的意思就是请别人帮忙做一些事情。读完文章之后你会发现,委托是指向一个方法的指针,通过指定一个委托名称,即可通过委托来调用方法。实际上与字面意思差不多。

  什么情况下适合使用委托

  首先通过一个例子来说明什么情况下使用委托。

  以学生值日为例,值日生要做开门、擦黑板的工作。对应的方法形式如下:

public void OpenDoor(){
    //开门具体实现
    ...
}

public void CleanBlackBoard(){
    //擦黑板具体实现
    ...
}

  下面再写一个值日的方法,要完成上述这些工作:

public void OnDuty(){
    OpenDoor();
    CleanBlackBoard();}

  这样,值日生开门、擦黑板的工作就实现了。但是这种方法的扩展性和灵活性不是很好,假设现在值日生工作又增加了,需要关灯,方法如下:

public void TurnOffLight(){
    //关灯的具体逻辑
    ...
}

  我们就需要修改OnDuty方法:

public void OnDuty(){
    OpenDoor();
    CleanBlackBoard();
    TurnOffLight();
}

  上面的代码没有使用委托,但是还是实现了需要完成的工作。很容易发现,开门、擦黑板、关灯,虽然他们的方法名称不同,但是它们具有相同的“形式”——它们都不获取参数,也都没有返回值(void),这正是委托可以发挥作用的时候。使用于这种形式匹配的一个委托。就可以引用任何工作的方法。我们可以声明如下的一个委托:

public delegate void DoSomeWorkDelegate();

 注意:

  • 声明委托时,要使用 delegate 关键字。
  • 委托定义了它能引用的方法的“形式”。你要指定返回类型(本例是void)、委托名称(DoSomeWorkDelegate)以及任何参数(本例无参数)。

  定义好委托之后,就可以创建它的一个实例,并使用 “+=”操作符,让这个实例引用一个相匹配的方法。代码如下:

public delegate void DoSomeWorkDelegate();
public DoSomeWorkDelegate doSomeWork;

doSomeWork+=OpenDoor;

  上例中的方法很简单,既没有返回值,又没有参数。目的只是了解在何种情况下使用委托。下面通过一个带参数的例子来说明继续说明如何使用委托。

  如何使用委托

  此处以听课为例,为了对比,同样先不使用委托,先上代码:

public void AttendClass(string name) {
    // 做某些额外的事情,比如初始化之类,此处略
    ChineseClass(name);
}
public void ChineseClass(string name) {
     //具体听语文课的逻辑
     ...  
}

  上面这段代码表示。AttendClass 表示听课,当我们传递代表学生姓名 name 参数,比如说“Jhon”,进去的时候,在这个方法中,将调用ChineseClass 方法,再次传递 Jhon 参数, 表示Jhon 参加了语文课。

  假设现在开设了数学课,我们需要添加新的方法:

public void MathClass(string name){
    //具体听数学课的逻辑    ...  
}

  这时候 AttendClass 也要修改,在此之前先定义一个枚举用作判断:

public enum Subject{
    Chinese,Math
}

public void AttendClass(string name, Subject sub){
    //做某些额外的事情,比如初始化之类,此处略
    swith(sub){
        case Subject.Chinese:           ChineseClass(name);
           break;
       case Subject.Math:
           MathClass(name);
           break;
    }
}

  OK,现在问题解决了,诚如前一个例子所说,假如现在又要增加英语课,体育课,就不得不反复修改枚举和 AttendClass 。可见程序的可拓展性很不好。

如前例,对于上课的方法,不论是什么课,它们都具有相同的“形式”,这样我们即可使用委托。

在考虑如何使用委托之前,我们先看看 AttendClass  的方法签名:

public void AttendClass(string name, Subject sub)

  我们仅看 string name,在这里,string 是参数类型,name 是参数变量,当我们给字符串那么赋值时,我们可以在方法体内对这个name进行其他操作.现在假如现在让 AttendClass() 方法接受一个参数变量,这个变量可以代表另一个方法,当给这个变量赋值为 ChineseClass 时,它代表 ChineseClass() 这个方法,当给它赋值为 MathClass 时,他代表MathClass() 这个方法。就是说让用一个参数来代方法。比如,把这个参数命名为,SubClass ,然后我们在方法体内,就像使用参数一样使用SubClass。由于 SubClass 代表着一个方法,它的使用方式应该和它被赋值的方法(比如 ChineseClass)是一样的,比如

SubClass(string name)

  按照这样的思路这样,AttendClass(),就应该是这个样子的:

public void AttendClass(string name,*** SubClass){
    SubClass(name);
}

  注意到 *** ,这个位置通常放置的应该是参数的类型,现在就出现了一个问题:这个代表着方法的SubClass参数应该是什么类型的?

  如果我说答案就是委托。它定义了 SubClass 方法的种类,也就是 SubClass 方法参数的类型。如同 string 决定了AttendClass() 这个方法的第一个参数的类型一样,委托决定 AttendClass 方法中的第二个参数的类型。

  本例中委托的定义如下:

public delegate void ClassDelegate(string name);

  现在再次改动 AttendClass 方法:

public void AttendClass(string name,ClassDelegate SubClass){
    SubClass(name);
}

  委托ClassDelegate出现的位置与 string 相同,string是一个类型,那么ClassDelegate 应该也是一个类型,或者叫类(Class)。但是委托的声明方式和类却完全不同。那么本例完整有关上课的类的代码如下:

public delegate void ClassDelegate(string name);
public class OnClass{
        //其它代码
        ...
        public void AttendClass(string name,ClassDelegate SubClass){
                SubClass(name);
        }

        public void ChineseClass(string name){
                ...
        }
        public void MathClass(string name){
                ...
        }
}

  由此可见,委托是一种类型。它定义了方法的类型,实际上就是把方法当做变量。(注意与指针的区别)

  那么我们如何使用它呢,代码如下:

private OnClass objOnClass;
objOnClass=new OnClass();
string name1="xiaoming";
string name2="小张";
objOnClass.AttendClass(name1,ChineseClass);
objOnClass.AttendClass(name2,MathClass);

  上面的代码表示 xiaoming 参加了语文课,小张参加了数学课。既然委托是一种类型。那么我们也可以以下面的方式来运用委托

ClassDelegate delegate1,delegate2;
delegate1=objOnClass.ChineseClass;
delegate2=objOnClass.MathClass;
string name1="xiaoming";
string name2="小张";
objOnClass.AttendClass(name1,delegate1);
objOnClass.AttendClass(name2,delegate2);

  还可以将多个方法付给同一个委托,下面的代码表示小张既参加了语文课,又参加了数学课

ClassDelegate delegate3;
delegate3=objOnClass.ChineseClass;
delegate3+=objOnClass.MathClass;
string name1="小张";
objOnCLass.AttendClass(name1,delegate3);

  可以看到,为委托绑定方法我们使用 重载了的 += 这个符号。如果将上面的代码中第一次绑定方法 delegate3=ChineseClass 改为 delegate3+=ChineseClass,将会出现“使用了未赋值的局部变量”的错误。我们可以像下面这样使用委托,给它初始化赋值

ClassDelegate delegate3=new ClassDelegate(objOnClass.ChineseClass);
string name1="xiaoming";
objOnClass.AttendClass(name1,delegate3);

  我们甚至可以不用AttendClass这个方法,像下面这样来使用委托

ClassDelegate delegate3=new ClassDelegate(objOnClass.ChineseClass);
string name1="xiaoming";
delegate3(name1);

  所以,可以将多个方法绑定到一个委托上面,调用的时候依次执行。

学习和理解C#的委托的更多相关文章

  1. SQL Server 学习博客分享列表(应用式学习 + 深入理解)

    SQL Server 学习博客分享列表(应用式学习 + 深入理解) 转自:https://blog.csdn.net/tianjing0805/article/details/75047574 SQL ...

  2. JDK学习---深入理解java中的HashMap、HashSet底层实现

    本文参考资料: 1.<大话数据结构> 2.http://www.cnblogs.com/dassmeta/p/5338955.html 3.http://www.cnblogs.com/d ...

  3. JDK学习---深入理解java中的LinkedList

    本文参考资料: 1.<大话数据结构> 2.http://blog.csdn.net/jzhf2012/article/details/8540543 3.http://blog.csdn. ...

  4. python基础知识的学习和理解

    参考链接:https://github.com/yanhualei/about_python/tree/master/python_learning/python_base   python基础知识笔 ...

  5. 【log4j】的学习和理解 + 打印所有 SQL

    log4j 1.2 学习和理解 + 打印所有 SQL 一.基本资料 官方文档:http://logging.apache.org/log4j/1.2/manual.html(理解基本概念和其他) lo ...

  6. 学习和理解C#中的事件

    注:本文系学习笔记. 上一篇文章记录了我对C#中委托的理解.委托实际上是一种类型.可以将一个或多个方法绑定到委托上面,调用委托时,一次执行委托上面绑定的方法.本文要讲述的事件实际上和委托有很深的“感情 ...

  7. STM32学习笔记(四) RCC外设的学习和理解

    RCC时钟模块并不好理解,初次接触我也是一头雾水,而且我真正掌握它的时候也比较晚,是我在学习uC/os-II,需要分析时钟时才有了深刻认识.但在学习中我却一定要把放在了前列,因为这是整个嵌入式最重要的 ...

  8. PHP面向对象三大特点学习(充分理解抽象、封装、继承、多态)

    PHP面向对象三大特点学习 学习目标:充分理解抽象.封装.继承.多态   面象对向的三大特点:封装性.继承性.多态性 首先简单理解一下抽象:我们在前面定义一个类的时候,实际上就是把一类事物共有的属性和 ...

  9. 《C#高级编程》学习笔记------C#中的委托和事件(续)

    本文转载自张子阳 目录 为什么要使用事件而不是委托变量? 为什么委托定义的返回值通常都为void? 如何让事件只允许一个客户订阅?(事件访问器) 获得多个返回值与异常处理 委托中订阅者方法超时的处理 ...

随机推荐

  1. PHP-CGI, FastCGI, PHP-FPM的关系和区别

    Web server(apache, nginx) 接受到一个php请求后要解析php文件, 怎么解析呢, web server是C语言写的, 所以需要一个协议, 一个php解释器, 也就是CGI. ...

  2. PHP 如何阻止用户上传成人照片或者裸照

    在这份教程中,我们将会学习到如何阻止用户通过PHP上传成人照片或者裸照. 示例   下载 我在phpclasses.org上面偶然发现一个很有用的,由Bakr Alsharif开发的可以帮助开发者基于 ...

  3. Git教程(9)集中式工作方式常用的设计分支的方案

    Git是一个复杂的版本管理系统,管理代码有很多工作方式,如集中式,管理者式,司令/副官式 本文是假设选用集中式工作方式时,设计分支的方案. 中小型项目: 维护两个长期分支,分别是master 和 de ...

  4. Windows Embedded Compact 2013升级:VS2013也能编译

    IT之家(www.ithome.com):Windows Embedded Compact 2013升级:VS2013也能编译 今天,微软为Windows Embedded Compact 2013送 ...

  5. Scrapy在win7 32位的安装及依赖包

    Scrapy,一个网络爬虫的框架,首先第一步肯定是安装. 参考网上的文章. 安装过程中需要用到pip工具,请自行安装. 1.安装python 这个是必须的,既然都用到scrapy了,肯定已经安装了py ...

  6. eclipse 点击 open Implementation就退出eclipse

    输入法不对.. 切换到纯英文的输入法. 微软自带的那个..  我电脑上也这样. 现在好了 (安装谷歌输入法貌似存在这个问题)

  7. mysql 读取硬盘数据

    innodb 的最小管理单位是页 innodb的最小申请单位是区,一个区 1M,内含64个页,每个页16K ,即 64*16K=1M, 考虑到硬盘局部性,每次读取4个区,即读4M的数据加载至内存 线性 ...

  8. RazorEngine(未解决,留底)

    TemplateServiceConfiguration templateConfig = new TemplateServiceConfiguration { BaseTemplateType = ...

  9. BZOJ3258: 秘密任务

    题解: 其实就是一个简单的最小割判断是否唯一解... 可是我写了一上午还没过...T_T 把1-n的最短路上的边提出来做最小割. 然后从s,t分别bfs判断必须在某个割的点.如果有的点没有被bfs到, ...

  10. (转)理解OAuth 2.0

    转自:http://www.ruanyifeng.com/blog/2014/05/oauth_2_0.html OAuth是一个关于授权(authorization)的开放网络标准,在全世界得到广泛 ...