C#:类的成员--事件
鸿门宴上项羽摔杯为号,埋伏两侧的刀斧手就会杀将而来,将刘邦一干人等就地伏法。这就是一个事件模型。
事件模型:事件拥有者、事件、事件响应者、响应事件的处理方法、事件订阅;当某种情况发生时,触发事件。
- 以鸿门宴摔杯为号的故事为例:
1. 事件拥有者:项羽
2. 事件:摔杯子
3. 事件参数(重要但非必须):谁摔了杯子(对于杀手来说,管你3721,听过杯子摔了就是干。)
4. 事件响应者:刀斧手
5. 事件订阅:项羽对刀斧手说:“明日设宴,孤以摔杯为号,尔等即刻行事!”
6. 事件触发:项羽看劝降不成,勃然大怒,抓起杯子摔倒地上。(事件发生是需要触发的,否则事件永远不会发生,事件响应方法也永远不会执行。)
下面通过自定义事件,来模拟摔杯为号的事件模型:
声明事件的第一种方式
class Program
{
static void Main(string[] args)
{
XiangYu zhugong = new XiangYu();
ShaShou daofushou = new DaoFuShou();
ShaShou gongjianshou = new GongjianShou();
//6 杀手订阅主公的摔杯事件
zhugong.ShuaiBeiziEvent += new EventHandler<BeiziEventArgs>(daofushou.Action);
zhugong.ShuaiBeiziEvent += gongjianshou.Action;
//7 触发事件--- 谈判中(项羽可能随时会摔杯子)
zhugong.Quanxiang();
Console.ReadKey();
}
}
//1 事件携带的附加信息
class BeiziEventArgs : EventArgs
{
//杯子材质
public string Metarials { get; set; }
public BeiziEventArgs(string meterials)
{
this.Metarials = meterials;
}
}
//2 定义事件拥有者
class XiangYu
{
private EventHandler<BeiziEventArgs> eventHandler;
//3 定义摔杯事件
public event EventHandler<BeiziEventArgs> ShuaiBeiziEvent
{
add
{
eventHandler += value;
}
remove
{
eventHandler -= value;
}
}
public void Quanxiang()
{
var ss = "你可以按下s来触发摔杯事件";
Console.WriteLine(ss);
do
{
Console.WriteLine("宴会进行中...");
ss = Console.ReadLine();
} while ("s"!=ss);
Console.WriteLine("项羽怒了 抄起酒杯摔了下去");
if (this.eventHandler!=null)
{
BeiziEventArgs attachinfo = new BeiziEventArgs("葡萄美酒夜光杯");
eventHandler.Invoke(this, attachinfo);
}
}
}
//4 定义事件响应者
abstract class ShaShou
{
//5 在响应者中 定义事件处理方法
public abstract void Action(object sender, BeiziEventArgs e);
}
class DaoFuShou : ShaShou
{
//刀斧手
public override void Action(object sender, BeiziEventArgs e)
{
Console.WriteLine($"{e.Metarials}被摔了,进去拿人");
}
}
class GongjianShou : ShaShou
{
//弓箭手
public override void Action(object sender, BeiziEventArgs e)
{
Console.WriteLine($"{e.Metarials}被摔了,准备放箭");
}
}
运行结果:
代码分析:
- 这是一个标准的 事件位于事件拥有者中、事件处理方法位于事件响应者中的实现方式:
- 从事件的声明方式上可以看出,事件是类的一个成员;不要把事件误认为是一种类型;
- 第六步订阅事件的时候,可以使用委托实例、也可以使用方法名,这是一个语法糖;
声明事件的第二种方式:使用事件声明语法糖,来完成上面的功能,在XiangYu类型改动如下:
class XiangYu
{
//3 定义摔杯事件
public event EventHandler<BeiziEventArgs> ShuaiBeiziEvent;
public void Quanxiang()
{
var ss = "你可以按下s来触发摔杯事件";
Console.WriteLine(ss);
do
{
Console.WriteLine("宴会进行中...");
ss = Console.ReadLine();
} while ("s"!=ss);
Console.WriteLine("项羽怒了 抄起酒杯摔了下去");
if (this.ShuaiBeiziEvent!=null)
{
BeiziEventArgs attachinfo = new BeiziEventArgs("葡萄美酒夜光杯");
ShuaiBeiziEvent.Invoke(this, attachinfo);
}
}
}
为什么说 public event EventHandler
其实从反编译出来的代码可以看出,这两个方法的详细实现:
分析以上内容后我们发现,事件声明语法糖,是编译器以一种安全的方式为我们隐去了第一种方式的步骤,所以我们推荐第二种方式,但要知其然知其所以然;另外,事件是委托的包装器。他包装的这个委托内部维护了一个委托链。
下面是"摔杯为号"故事的详细关系图:
本篇中并没有定义一种新的委托类型,而是使用了.NET为我们提供的EventHanler委托类型,为的是不做额外的工作(也不建议无脑定义委托类型,因为.NET为我们提供的委托类型大部分都够用了);若真的需要定义一种委托类型来约束事件处理方法的话,委托类型名要以EventHadnler结尾,如:ClickEventHandler;事件模式更多是用于客户端程序中的,因为他们是基于事件驱动的;事件用的多,而自己定义的情况很少,但是在我们使用事件前了解清楚事件的原理是十分重要的。
如本文的标题一样:事件是类的成员;类的三大成员:属性、方法、事件。以上便是对事件的总结,记录下来以便以后查阅。