分分钟用上C#中的委托和事件
每一个初学C#的程序猿,在刚刚碰到委托和事件的概念时,估计都是望而却步,茫然摸不到头脑的。百度一搜,关于概念介绍的文章大把大把的,当然也不乏深入浅出的好文章。可看完这些文章,大多数新手,估计也只是信心满满的觉得自己看懂了,一旦要在自己的程序中用上委托和事件,就傻眼了,根本不知道什么情况下该用,也不知道具体怎么用。
本文没有任何概念介绍,旨在帮助同学们快速用上委托和事件,进入C#的更高境界。
- 首先,我们需要知道,到底在什么情况下必须使用委托和事件呢?
请看下面的场景:首领A要搞一场鸿门宴,吩咐部下B和C各自带队埋伏在屏风两侧,约定以杯为令:若左手举杯,则B带队杀出;若右手举杯,则C带队杀出;若直接摔杯,则B和C同时杀出。B和C袭击的具体方法,首领A并不关心。(完整代码在全文最后。)
有脑子反应快的同学,马上说,这个逻辑,用一个IF条件判断语句不就搞定了吗:
if(左手举杯)
{
B带队杀出;
}
else if(右手举杯)
{
C带队杀出;
}
else if(直接摔杯)
{
B带队杀出;
C带队杀出;
}
else
{
按兵不动
}
如果真这么简单,那我还写这篇文章干嘛呢。请这位同学认真想想:首领A会什么时候发出信号呢?估计连他自己都不知道。那么上面这段伪码,你打算插入到哪里呢?难道B和C就一直不停的用While循环做上述判断吗?显然不合适吧。更重要的,如果扩展一下,部下不止B和C,而是B、C、D、E等等,每个人都有自己不同的响应领导号召的方法,那么对首领A而言,要分别去跟每个部下打交道,上述的伪码,会进行无尽的扩展和完善,显然也不是我们所希望的。
正确的逻辑应该是,B和C不管宴席上发生的任何其他事情,只等首领发出举杯或者摔杯的信号,一旦首领A发出信号,相当于通知了所有部下,凡是约定好的部下,都立马各自行动!
我们定义三个类,分别模拟首领A和部下B、C:
/// <summary>
/// 首领A
/// </summary>
public class A
{ }
/// <summary>
/// 部下B
/// </summary>
public class B
{ }
/// <summary>
/// 部下C
/// </summary>
public class C
{ }
首领A的类有举杯和摔杯两种方法,其中,举杯方法带有一个形参,用于传递是左手举杯,还是右手举杯。部下B和C的类,各自有一种攻击方法。
/// <summary>
/// 首领A
/// </summary>
public class A
{
/// <summary>
/// 举杯
/// </summary>
/// <param name="hand">手:左、右</param>
public void Raise(string hand)
{
Console.WriteLine("首领A{0}手举杯", hand);
}
/// <summary>
/// 摔杯
/// </summary>
public void Fall()
{
Console.WriteLine("首领A摔杯");
}
}
/// <summary>
/// 部下B
/// </summary>
public class B
{
/// <summary>
/// 攻击
/// </summary>
public void Attack()
{
Console.WriteLine("部下B发起攻击");
}
}
/// <summary>
/// 部下C
/// </summary>
public class C
{
/// <summary>
/// 攻击
/// </summary>
public void Attack()
{
Console.WriteLine("部下C发起攻击");
}
}
至此,三个独立的类,构造完毕。那么,怎么让部下B和C,根据首领的暗语才去相应的行动呢?
我们需要在首领A类中,采用一种方法,把其意图传递出去。在首领A类之前,分别定义一个带形参的举杯委托RaiseEventHandler和一个不带形参的摔杯委托FallEventHandler。命名规则是在准备传递的方法名后加上EventHandler,不要问我为什么,介绍为什么的文章多得是,大家大可以去查阅。本文的宗旨是,教会大家如何快速的使用委托和事件。当然,你也可以不遵循此命名规则,对程序的运行没有任何影响,只是会增加自己或他人日后阅读代码的难度。
delegate void RaiseEventHandler(string hand);
delegate void FallEventHandler();
然后,在首领A类中,定义两个事件。这个类似于大家在WinForm中拖放一个Button后,双击该Button,即可编辑其事件。
/// <summary>
/// 首领A举杯事件
/// </summary>
public event RaiseEventHandler RaiseEvent;
/// <summary>
/// 首领A摔杯事件
/// </summary>
public event FallEventHandler FallEvent;
好了,做完上面两个步骤后,就可以在首领A的举杯和摔杯方法中,调用上述两个事件了。这样,如果B和C中订阅该事件,便可自动执行了。
/// <summary>
/// 举杯
/// </summary>
/// <param name="hand">手:左、右</param>
public void Raise(string hand)
{
Console.WriteLine("首领A{0}手举杯", hand);
// 调用举杯事件,传入左或右手作为参数
if (RaiseEvent!=null)
{
RaiseEvent(hand);
}
}
/// <summary>
/// 摔杯
/// </summary>
public void Fall()
{
Console.WriteLine("首领A摔杯");
// 调用摔杯事件
if (FallEvent!=null)
{
FallEvent();
}
}
同样,不要问为什么这样写。按照上述操作即可。一旦用熟这种方法,体会其中含义,自然就明白为何这么写了。
在部下B和C心中,必须存在首领A,才能执行A的暗示吧。所以,在B类和C类中,需要声明一个A,该声明可以通过B和C的构造函数进行实例化。实例化之后,便可在类B和类C中订阅类A的事件了。
完整代码如下:
class Program
{
static void Main(string[] args)
{
A a = new A(); // 定义首领A B b = new B(a); // 定义部下B C c = new C(a); // 定义部下C // 首领A左手举杯
a.Raise("左"); // 首领A右手举杯
//a.Raise("右"); // 首领A摔杯
//a.Fall(); Console.ReadLine();
// 由于B和C订阅了A的事件,所以无需任何代码,B和C均会按照约定进行动作。
}
} /// <summary>
/// 首领A举杯委托
/// </summary>
/// <param name="hand">手:左、右</param>
public delegate void RaiseEventHandler(string hand);
/// <summary>
/// 首领A摔杯委托
/// </summary>
public delegate void FallEventHandler();
/// <summary>
/// 首领A
/// </summary>
public class A
{
/// <summary>
/// 首领A举杯事件
/// </summary>
public event RaiseEventHandler RaiseEvent;
/// <summary>
/// 首领A摔杯事件
/// </summary>
public event FallEventHandler FallEvent; /// <summary>
/// 举杯
/// </summary>
/// <param name="hand">手:左、右</param>
public void Raise(string hand)
{
Console.WriteLine("首领A{0}手举杯", hand);
// 调用举杯事件,传入左或右手作为参数
if (RaiseEvent!=null)
{
RaiseEvent(hand);
}
}
/// <summary>
/// 摔杯
/// </summary>
public void Fall()
{
Console.WriteLine("首领A摔杯");
// 调用摔杯事件
if (FallEvent!=null)
{
FallEvent();
}
}
}
/// <summary>
/// 部下B
/// </summary>
public class B
{
A a; public B(A a)
{
this.a = a;
a.RaiseEvent += new RaiseEventHandler(a_RaiseEvent); // 订阅举杯事件
a.FallEvent += new FallEventHandler(a_FallEvent); // 订阅摔杯事件
}
/// <summary>
/// 首领举杯时的动作
/// </summary>
/// <param name="hand">若首领A左手举杯,则B攻击</param>
void a_RaiseEvent(string hand)
{
if (hand.Equals("左"))
{
Attack();
}
} /// <summary>
/// 首领摔杯时的动作
/// </summary>
void a_FallEvent()
{
Attack();
} /// <summary>
/// 攻击
/// </summary>
public void Attack()
{
Console.WriteLine("部下B发起攻击,大喊:猛人张飞来也!");
}
}
/// <summary>
/// 部下C
/// </summary>
public class C
{
A a;
public C(A a)
{
this.a = a;
a.RaiseEvent += new RaiseEventHandler(a_RaiseEvent); // 订阅举杯事件
a.FallEvent += new FallEventHandler(a_FallEvent); // 订阅摔杯事件
}
/// <summary>
/// 首领举杯时的动作
/// </summary>
/// <param name="hand">若首领A右手举杯,则攻击</param>
void a_RaiseEvent(string hand)
{
if (hand.Equals("右"))
{
Attack();
}
} /// <summary>
/// 首领摔杯时的动作
/// </summary>
void a_FallEvent()
{
Attack();
}
/// <summary>
/// 攻击
/// </summary>
public void Attack()
{
Console.WriteLine("部下C发起攻击,一套落英神掌打得虎虎生威~");
}
}
分分钟用上C#中的委托和事件的更多相关文章
- 分分钟用上C#中的委托和事件之窗体篇
上次以鸿门宴的例子写了一篇名为<分分钟用上C#中的委托和事件>的博文,旨在帮助C#初学者迈过委托和事件这道坎,能够用最快的速度掌握如何使用它们.如果觉得意犹未尽,或者仍然不知如何在实际应用 ...
- C#中的委托和事件(下篇)
上次以鸿门宴的例子写了一篇博文,旨在帮助C#初学者迈过委托和事件这道坎,能够用最快的速度掌握如何使用它们.如果觉得意犹未尽,或者仍然不知如何在实际应用中使用它们,那么,这篇窗体篇,将在Winform场 ...
- C# 中的委托和事件
觉得这篇文章写的非常好,大神之作,由简入繁,对我这种初学者来说帮忙很大,特此留存下. 摘自:http://tracefact.net/CSharp-Programming/Delegates-and- ...
- C# 中的委托和事件(转)
引言 委托 和 事件在 .Net Framework中的应用非常广泛,然而,较好地理解委托和事件对很多接触C#时间不长的人来说并不容易.它们就像是一道槛儿,过了这个槛的人,觉得真是太容易了,而没有过去 ...
- C# 中的委托和事件(转载)
引言 委托 和 事件在 .Net Framework中的应用非常广泛,然而,较好地理解委托和事件对很多接触C#时间不长的人来说并不容易.它们就像是一道槛儿,过了这个槛的人,觉得真是太容易了,而没有过去 ...
- 【转】C# 中的委托和事件
阅读目录 C# 中的委托和事件 引言 将方法作为方法的参数 将方法绑定到委托 事件的由来 事件和委托的编译代码 委托.事件与Observer设计模式 .Net Framework中的委托与事件 总结 ...
- 《C#高级编程》学习笔记------C#中的委托和事件(续)
本文转载自张子阳 目录 为什么要使用事件而不是委托变量? 为什么委托定义的返回值通常都为void? 如何让事件只允许一个客户订阅?(事件访问器) 获得多个返回值与异常处理 委托中订阅者方法超时的处理 ...
- c#中的委托和事件(转)
引言 委托 和 事件在 .Net Framework中的应用非常广泛,然而,较好地理解委托和事件对很多接触C#时间不长的人来说并不容易.它们就像是一道槛儿,过了这个槛的人,觉得真是太容易了,而没有过去 ...
- C#中的委托和事件(续)
转自张子阳的博客http://www.tracefact.net/CSharp-Programming/Delegates-and-Events-Advanced.aspx 引言 如果你看过了 C#中 ...
随机推荐
- JavaScript通过ID获取元素坐标
JavaScript通过ID获取元素坐标 function getElementPos(elementId) { var ua = navigator.userAgent.toLowerCase ...
- *** $CI =& get_instance() 用法:关于CodeIgniter中get_instance() 函数
使用场景: 注意 get_instance 的使用场景,这个方法并不是用在控制器中的.而是用在控制器外面,比如类库中,想操作 CI 超级对象的时候,超级对象实际上就是当前控制器的实例. 你随便下个CI ...
- Android基础学习第一篇—Project目录结构
写在前面的话: 1. 最近在自学Android,也是边看书边写一些Demo,由于知识点越来越多,脑子越来越记不清楚,所以打算写成读书笔记,供以后查看,也算是把自己学到所理解的东西写出来,献丑,如有不对 ...
- java反射详解
本篇文章依旧采用小例子来说明,因为我始终觉的,案例驱动是最好的,要不然只看理论的话,看了也不懂,不过建议大家在看完文章之后,在回过头去看看理论,会有更好的理解. 下面开始正文. [案例1]通过一个对象 ...
- SQL谜题(楼层谜题)
Multiple DwellingsBaker, Cooper, Fletcher, Miller and Smith live on different floors of an apartment ...
- PDA固定资产条码管理系统软件-解决固定资产实物清查的瓶颈问题,大大提高清查效率
固定资产管理系统是企业信息化管理中的一个重要组成部分,固定资产具有价值高,使用周期长.使用地点分散.管理难度大等特点.一个企业的良性发展,避免不了的要涉及到企业资产的有效管理.对于那些技术装备密集型的 ...
- 【转】UML图与软件开发过程那点关系
首先,软工文档, 软工文档,也就是计划,设计,描述,使用软件的一些文件,它最大的特点就是固定不变,用来给不同的人和计算机来阅读.在期间,文档起到了桥梁的作用,看这张图很形象: 在这里在看一下国家统一规 ...
- 1.2 控制器 view 的创建和加载
本文并非最终版本,如有更新或更正会第一时间置顶,联系方式详见文末 如果觉得本文内容过长,请前往本人 “简书” 加载优先顺序: 1.用系统的loadView方法创建控制器的视图 2.如果指定 ...
- bzoj 4330: JSOI2012 爱之项链
听说这题不公开.. 那就不贴题意了 首先要用burnside引理求出戒指的种数,那么对于一个顺时针旋转$k$个位置的置换就相当于连上一条$(i,(i+k)%R)$的边,每个环颜色必须相同 环的个数为$ ...
- BZOJ 2152 & 点分治
Description: 只是打法法塔前测试一下板子 Code: /*================================= # Created time: 2016-04-20 14:3 ...