链方法[C# 基础知识系列]专题三:如何用委托包装多个方法——委托链

最近研究链方法,稍微总结一下,以后继续补充:

    

弁言:

    

 上一专题分析了下编译器是如何来翻译委托的,从中间语言的角度去看委托,希望可以帮助大家进一步的理解委托,然而之前的分析都是委托只是封装一个方法,那委托能不能封装多个方法呢?因为生活中经常会听到,我代表大家的意见等这样的谈话,既然委托也是一个代表,那他如果只能代表一个人,那他的魅力就不是很大了吧,所以我们就会委托能不能代表多个方法的? 答案是可以的,这就是本专题要讲的内容——委托链,委托链也是一个委托,只是因为它是把多个委托链在一起,所以我们就以委托链来这么称呼它的。

    

 

    

一、到底什么是委托链

    

我们平常实例化委托对象时都是绑定一个方法的, 前一个专题分析的委托也是包装了一个方法的, 用后面的例子就是委派律师的只有一个人,也就是当事人只有一个的,但是现实生活中明显不是这样的,在讼事的时候律师可以同时接多个案子,也是接收多个当时人的委派,这样,该律师就与多个当事人绑定在一起了, 须要懂得多个当事人的案件情况的。其实这就是生活中的委托链,此时这位律师不仅仅是一个人的代表律师了,而是多个当事人的律师。生活中的委托链和C#中的委托链很类似的,当初就说说C#中的委托链究竟是个什么的?

    

首先委托链就是一个委托,所以大家不要看到委托链感觉又是什么C#中的新特性的,然而要把多个委托链在一起,就必须存储多个委托的引用,那委托链对象是在哪里存储多个委托的引用的呢?还记得我们上一专题中,我们分析的委托类型有三个非公共字段的吗?这三个字段是——_target,methodPtr 和_invocationList,至于这三个字段具体代表什么大家可以查看我的上一专题的文章,然而_invocationList 字段正是存储多个委托引用的地方的。

    

为了更好的解释_invocationList是如何来存储委托引用的,下面先看一个委托链的例子和运行结果,然后再分析原因:

    

 

using System;namespace DelegateTest{public class Program{// 声明一个委托类型,它的实例引用一个方法// 该方法回去一个int 参数,返回void类型public delegate void DelegateTest(int parm);public static void Main(string[] args){// 用静态方法来实例化委托DelegateTest dtstatic = new DelegateTest(Program.method1);// 用实例方法来实例化委托DelegateTest dtinstance = new DelegateTest(new Program().method2);// 隐式调用委托dtstatic(1);// 显式调用Invoke方法来调用委托dtinstance.Invoke(1);// 隐式调用委托dtstatic(2);// 显式调用Invoke方法来调用委托dtinstance.Invoke(2);Console.Read();}private static void method1(int parm){Console.WriteLine("调用的是静态方法,参数值为:" + parm);}private void method2(int parm){Console.WriteLine("调用的是实例方法,参数值为:" + parm);}}}

    

运行结果:

    

链和方法

    

下面就来分析下为什么会涌现这样的结果的:

    

一开始我们实例化了两个委托变量,如下代码:

// 用静态方法来实例化委托DelegateTest dtstatic = new DelegateTest(Program.method1);// 用实例方法来实例化委托DelegateTest dtinstance = new DelegateTest(new Program().method2);

    

委托变量dtstatic和dtinstance引用的委托对象的初始状态如下图:

    

链和方法

    

然后我们定义了一个委托类型的引用变量delegatechain,刚开始它没有任何委托对象,是一个空引用,当我们执行下面的一行代码时,

delegatechain = (DelegateTest)Delegate.Combine(delegatechain, dtstatic);

    

Combine方法发明试图合并的是null和dtstatic,在内部,Combine直接返回dtstatic中的对象,此时delegatechain和dtstatic变量引用的都是同一个委托对象,如下图所示:

    

链和方法

    

为了演示委托链,我们通过代码在再添加一个委托,此时就再调用了Combine方法,代码如下:

delegatechain = (DelegateTest)Delegate.Combine(delegatechain, dtinstance);

    

这时候,Combine方法发明delegatechain已引用了一个委托对象了(此时已引用了destatic引用的委托对象了),所以Combine会结构一个新的委托对象(这一点很想String.Concat,我们简略的使用是通过+操作符把两个字符串连接起来,关于字符串的讨论大家可以参考我博客中的这篇文章http://www.cnblogs.com/zhili/archive/2012/06/25/String_StringBuilder.html),这个新的委托对象会对它的私有字段_target 和_methodPtr字段停止初始化,然后此时_invocationList字段初始化为引用了一个委托对象的数组,这个数组的第一个元素(下标为0)就是被初始化为引用包装了method1方法的委托,数组的二个元素被初始化为引用包装了method2方法的委托(也就是dtinstance引用的委托对象),最后delegaechain被设为引用新建的这个委托对象,下面是一个图,可以帮助大家理解委托链(也叫多播委托):

    

链和方法

每日一情理
这浓浓的母爱使我深深地认识到:即使你是一只矫健的雄鹰,也永久飞不出母爱的长空;即使你是一条扬帆行驶的快船,也永久驶不出母爱的长河!在人生的路上不管我们已走过多远,还要走多远,我们都要经过母亲精心营造的那座桥!

    

同样的情理,如果是添加第三个委托给委托链,进程也是和下面一样的, 此时又会新建一个委托对象,此时_invocationList字段会初始化为引用一个保存这三个委托对象数组,然而有人会问了——对于已引用了委托对象的委托类型变量调用Combine方法后会创建一个新的委托对象,然后对新的这个委托对象的三个字段停止重新初始化话,最后把之前的委托类型变量引用新创建的委托对象(这里就帮大家总结下委托链的创建进程),那之前的委托对象怎么办呢? 相信大部分人会有这个疑问的,这点和字符串的Concat方法很像,之前的委托对象和——invocationList字段引用的数组会被垃圾回收失落(正是因为这样,委托和字符串String一样是不可变的)。

    

注意:我们还可以调用Delegate的Remove方法从链中删除委托,如调用下面代码时:

delegatechain =(DelegateTest)Delegate.Remove(delegatechain,new DelegateTest(method1));

    

Remove方法被调用时,它会扫描delegateChain(第一个参数)所引用的委托对象内部维护的委托数组(如果对于委托数组为空的情况下调用Remove方法将不会有任何作用,就是不会删除任何委托引用,这里主要是说明扫描是从委托数组里停止扫描),如果找到delegateChain引用的委托对象的_target和_methodPtr字段

    

和第二个参数(新创建的委托)中的字段匹配的委托,如果删除之后数组中只剩下一个数据项时,就返回那个数据项(而不会去新建一个委托对象再初始化的,此时的_invocationList为null,而不是保存一个委托对象引用的数组了,具体可以Remove一个后调试看看的),如果此时数组中还剩余多个数据项,就新建一个委托对象——其中创建并初始化_invocationList数组(此时的数组引用的委托对象已少了一个了,因为用Remove方法删除了),并且,每次Remove方法调用只能从链中删除一个委托,而不会删除有匹配的_target和_methodPtr字段的所有委托(这个大家可以调试看看的)

    

 

    

二、如何对委托链中的委托调用停止控制

    

通过下面相信大家可以理解如何创建一个委托链对象的,但是从运行结果中还可以看出,每次调用委托链时,委托链包装的每一个方法都会次序被执行,如果委托链中被调用的委托抛出一个异常,这样链中的后续所有对象都不能被调用,并且如果委托的后面拥有一个非void的返回类型,则只有最后一个返回值会被保存,其他所有回调方法的返回值都会被舍弃,这就意味着其他所有操作的返回值都永久看不到的吗? 事实却不是这样的,我们可以通过调用Delegate.GetInvocationList方法来显式调用链中的每一个委托,同时可以添加一些自己的定义输出。

    

GetInvocationList方法返回一个由Delegate引用构成的数组,其中每一个数组都指向链中的一个委托对象。在内部,GetInvocationList创建并初始化一个数组,让数据的每一个元素都引用链中的一个委托,然后返回对该数组的一个引用。如果_invocatinList字段为null,返回的数组只有一个元素,该元素就是委托实例本身。下面就通过一个程序来演示下的:

namespace DelegateChainDemo
{class Program{// 声明一个委托类型,它的实例引用一个方法// 该方法回去一个int 参数,返回void类型public delegate string DelegateTest();static void Main(string[] args){// 用静态方法来实例化委托DelegateTest dtstatic = new DelegateTest(Program.method1);// 用实例方法来实例化委托DelegateTest dtinstance = new DelegateTest(new Program().method2);DelegateTest dtinstance2 = new DelegateTest(new Program().method3);// 定义一个委托链对象,一开始初始化为null,就是不代表任何方法(我就是我,我不代表任何人)DelegateTest delegatechain = null;delegatechain += dtstatic;delegatechain += dtinstance;delegatechain += dtinstance2;delegatechain =(DelegateTest)Delegate.Remove(delegatechain,new DelegateTest(method1));delegatechain = (DelegateTest)Delegate.Remove(delegatechain, new DelegateTest(new Program().method2));Console.WriteLine(Test(delegatechain));Console.Read();}private static string method1(){return "这是静态方法1";}private string method2(){throw new Exception("抛出了一个异常");}private string method3(){return "这是实例方法3";}// 测试调用委托的方法private static string Test(DelegateTest chain){if (chain == null){return null;}// 用这个变量来保存输出的字符串StringBuilder returnstring = new StringBuilder();// 获取一个委托数组,其中每一个元素都引用链中的委托Delegate[] delegatearray=chain.GetInvocationList();// 遍历数组中的每一个委托foreach (DelegateTest t in delegatearray){try{//调用委托获得返回值returnstring.Append(t() + Environment.NewLine);}catch (Exception e){returnstring.AppendFormat("异常从 {0} 方法中抛出, 异常信息为:{1}{2}", t.Method.Name, e.Message, Environment.NewLine);}}// 把结果返回给调用者return returnstring.ToString();}}
}

    

运行结果截图:

    

链和方法

    

从运行结果可以看出,此时我们可以获得每一个回调方法的返回值,并且可以加入一些自定义的返回值的(程序中加入了换行字符串),这样便可以对委托链中的每一个委托对象停止控制了,即使其中一个抛出异常,此时我们也可以停止捕获,而不会致使后续的委托对象不能被调用的问题。

    

 

    

三、总结下

    

本专题主要分析如何创建一个委托链以及对于创建一个委托链的进程停止了具体的分享,第二部分主要先指出了委托了一些局限性,然后通过调用GetInvocationList方法来返回一个委托数组,这样便可以通过遍历委托数组中的每一个委托来通知委托的调用进程,这样便可以对委托链的调用停止更多的控制的。到此本专题也就分析完了,通过这三个专题对委托的分析,相信大家会对委托有一个更深的理解,然后为什么要写三个专题来具体分析委托的呢? 主要是后面要分析的事件,Lambda表达式,Linq方面的内容都是和委托有关系的,所以更好的理解委托将是后面特性的一个基本,希望这些对大家有帮助,我将在下一个专题里头分析事件。

文章结束给大家分享下程序员的一些笑话语录: 程序员的愿望
  有一天一个程序员见到了上帝.上帝: 小伙子,我可以满足你一个愿望.程序员: 我希望中国国家队能再次打进世界杯.
  上帝: 这个啊!这个不好办啊,你还说下一个吧!
  程序员: 那好!我的下一个愿望是每天都能休息6个小时以上.
  上帝: 还是让中国国家打进世界杯.

--------------------------------- 原创文章 By
链和方法
---------------------------------

转载于:https://www.cnblogs.com/jiangu66/p/3153266.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/261491.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

51单片机lcd1602的简单实用

基础知识准备 lcd1602简介 图片来自b站斌哥单片机 需要关注的有以下几点: RS : 高数据低命令;RS 1时,lcd1602读写数据,RS0时,lcd1602读写命令R/W:高读低写;RW1时,单片机读取lcd160…

firedebug调试Jquery

不了解的同学先“点这里”看看什么是Firebug。简单来说,Firebug是Firefox上用来监视、编辑和调试站点的CSS、HTML、DOM和JavaScript的扩展工具。 我们先到Firebug的主页上下载最新的版本装上,下面开始调试: 1、查看、编辑HTML元素及其(转) 安…

linux下Bash编程until语句及格式化硬盘分区等编写脚本(十)

linux下Bash编程until语句及格式化硬盘分区等编写脚本(十)1.循环语句结构总结1.1.while语句当条件满足时,进入循环语句while 条件; do语句done1.2.until语句当条件不满足时,进入循环语句until 条件; do语句done1.3.for 变量 in 列表; do 循环体done1.4.f…

halcon区域腐蚀膨胀算子_Halcon算子

Halcon部分算子功能:*读取一张图像read_image(Image,C:/Users/Desktop/无标题.png)*画一个矩形生成区域draw_rectangle1(3600, Row1, Column1, Row2, Column2)*获得矩形区域gen_rectangle1(Rectangle, Row1, Column1, Row2, Column2)*区域内最大、最小、最大-最小灰…

51单片机数字钟的实现

作业 用51单片机完成一个数字钟的计时及校时功能。 1、设置8个数码管的显示缓冲区为DISPBUFF[8],并初始化为02,02,-,04,08,-,03,06. 2、定时器0每隔5ms刷新一次数码管显示缓冲区,并显示时间(中断来做) 3、定时器1负责精确定时,计时1s,每隔…

Google Map API 学习三

转载于:https://www.cnblogs.com/MyFlora/p/3154066.html

委托声明----委托实例化-----委托调用

定义和使用委托分三个步骤&#xff1a; 1、委托声明。2、委托实例化。3、委托调用。 一、定义委托定义委托的语法如下&#xff1a; <访问修饰符> delegate 返回类型 委托名(); 定义委托和定义方法很相似&#xff0c;委托没有具体的实现体&#xff0c;由关键字delegate声明…

慎用SELECT INTO复制表

很多时候我们习惯于用SELECTINTO 复制一个表或表结构&#xff0c;因为它方便&#xff0c;快捷&#xff0c;而且在某些情况下效率比INSERT INTO 效率要高一些。但是要注意&#xff1a; SELECT INTO 复制表或表结构的时候&#xff0c;只是得到了一个“外壳”&#xff0c;就像克隆…

mysql sql语句使用技巧

mysql更新数据限制limitmysql更新时&#xff0c;要更新记录中某个区间的数据&#xff0c;只能用WHERE条件来限制了&#xff0c;用LIMIT只能限制更新多少条&#xff01;测试如下&#xff1a;UPDATE products SET goods_number goods_number10 ORDER BY goods_id DESC LIMIT 5,1…

wingdows安装psutil_psutil模块安装指南(win与linux)

1、windows下psutil模块安装&#xff1a;https://pypi.python.org/packages/3.4/p/psutil/下载符合版本的软件包下载&#xff0c;然后安装即可。2、ubuntu下载psutil模块的tar包安装复制代码 代码示例:wget https://pypi.python.org/packages/source/p/psutil/psutil-2.1.3.tar…

信号的采样与插值重建(包含matlab)

实验任务 双频信号的采样和插值重建&#xff1a;对双频信号进行采样&#xff08;符合奈奎斯特低通采样定理&#xff09;&#xff0c;绘制波形和频谱&#xff1b;编制sinc函数插值函数&#xff0c;对采样进行恢复&#xff0c;绘制波形。 实验原理 matlab代码 clear all; clos…

dc概论之IO约束

版权声明&#xff1a;转载时请以超链接形式标明文章原始出处和作者信息及本声明http://bb2hh.blogbus.com/logs/39654476.html 注明&#xff1a;如需转载&#xff0c;请注明作者出处&#xff0c;谢谢&#xff5e;&#xff0c;Author&#xff1a;pythonlong以下根据资料和个人体…

对寄存器ESP和EBP的一些理解

PS&#xff1a;EBP是当前函数的存取指针。即存储或者读取数时的指针基地址&#xff1b;ESP就是当前函数的栈顶指针。每一次发生函数的调用&#xff08;主函数调用子函数&#xff09;时&#xff0c;在被调用函数初始时&#xff0c;都会把当前函数&#xff08;主函数&#xff09;…

配置FCKeditor_2.6.3+fckeditor-java-2.4

下载 http://www.fckeditor.net/download FCKeditor.Javahttp://sourceforge.net/project/showfiles.php?group_id75348&package_id129511 -1- copy fckeditor文件夹到项目的webRoot目录下 -2- 添加Jar包 slf4j-simple-1.5.0.jar slf4j-api-1.5.0.jar java-core-2.4-beta-…

普通调幅(AM)与抑制载波双边带调幅(DSB)matlab编程实现

实验任务 实现单频信号的AM、DSB调制&#xff0c;绘制调制前后的波形和频谱 实验原理 matlab代码 AM调制 clear all; ts 0.0025; %信号抽样时间间隔 fs 1/ts; %抽样频率 t 0:ts:10-ts; %时间向量 df fs/length(t); %fft的频率分辨率 msg2 cos(2*pi*2*t); Pn fft(msg2…

electron 读取文件夹内容_如何使用Electron Framework选择,读取,保存,删除或创建文件...

本文概述为了处理文件(CRUD)的生命周期, 我们将使用对话框和文件系统组件。对话框模块提供了用于显示本机系统对话框(例如打开文件或警报)的API, 因此Web应用程序可以提供与本机应用程序和Node.js文件系统相同的用户体验。加载所需的依赖项我们需要加载以下依赖项, 以执行我们要…

MySQL中MySQL X.X Command Line Client一闪而过的问题

问题介绍&#xff1a;我安装完MySQL(我安装的是5.5)后&#xff0c;使用MySQL 5.5 Command Line Client&#xff0c;每次点击&#xff0c;总是一闪而过。解决方法如下&#xff1a;首先进入cmd 切入MySQL的安装目录&#xff0c;然后切入 bin 目录 &#xff0c;输入mysqld-nt --sk…

4pam调制与解调(matlab实现)

实验原理 这是一种使用脉冲幅度调制技术的线路编码。PAM4信号有四个电压电平&#xff0c;每个幅度电平分别对应逻辑比特00、01、10和11。换言之&#xff0c;PAM4编码的每个符号由2个比特组成&#xff0c;它们对应一个电压电平&#xff0c;即幅度。 00对应-3,01对应-1,10对应1,…

WinForm开发,窗体显示和窗体传值相关知识总结

以前对WinForm窗体显示和窗体间传值了解不是很清楚最近做了一些WinForm开发,把用到的相关知识整理如下 A.WinForm中窗体显示显示窗体可以有以下2种方法&#xff1a;Form.ShowDialog方法 &#xff08;窗体显示为模式窗体&#xff09;Form.Show方法 &#xff08;窗体显示为无模式…

阿尔法蛋机器人tf卡_阿尔法蛋机器人哪款好适合几岁孩子,超能蛋早教机真实效果评测(价格309元)...

阿尔法蛋机器人是安徽淘云科技旗下的儿童智能机器人产品&#xff0c;作为早教机还是非常有名的&#xff0c;那么阿尔法蛋机器人哪款好呢&#xff1f;不同机器人的功能不同&#xff0c;价位也不等&#xff0c;适合的年龄段不同&#xff0c;这里推荐一款比较全面的阿尔法蛋超能蛋…