C#基础之事件(2)
在“C#基础之事件(1)”中已对事件有了一个大概,这里对事件进行更深入的学习。
本节按以下内容展开:
1.事件拥有者与事件响应者的关系;
2.事件订阅的多种写法;
3.事件的订阅和取消订阅;
4.多事件返回值的获取;
1.事件拥有者与事件响应者的关系
以下按照最常见到最不常见的关系列出来,并用示例展示:

结合这四副图,分析下内存关系。所谓“XX者"就是指一个对象,就是new class(),对象所包含的事件、事件处理方法都是对象的成员。有了这个认识,在此用A图,这个最常用的关系来写一段“伪代码”。
public class 拥有者类
{
public event 事件类型 事件成员变量; public void 某个方法(参数)
{
。。。
事件成员变量(参数);//此处是调用事件处理方法
。。。
}
} public class 响应者类
{
private 拥有者类 _拥有者对象; public 响应者类(拥有者类 拥有者对象)
{
this._拥有者对象=拥有者对象;
_拥有者对象.事件成员变量+=事件处理方法;
} public string 事件处理方法(参数)
{
。。。
}
}
由上面代码可知,响应者中包含拥有者时,可以通过构造方法传参形式包含进去,实际应用中不限于这种形式,还可以是new 一个拥有者对象、静态类等方式,只要是能包含进去的都是可行的。其它几个关系就更简单了。这些关系搞清楚了,才能“下笔如有神”。
下面是一个实际代码示例(网络资源非原创):
从program类中可以看到,通过调用事件拥有者dg的OnAlarm方法触发了事件,最终是调用了响应者host类的catch方法,与上图中的A的关系模型一样。
class Program
{
static void Main(string[] args)
{
Dog dg = new Dog();
Host ht = new Host(dg);
DateTime now = new DateTime(, , , , , );
DateTime end = new DateTime(, , , , , );
Console.WriteLine("时间快接近深夜0时~~~~");
while (now < end)
{
Console.WriteLine(now);
Thread.Sleep();
now = now.AddSeconds();
}
//午夜零点小偷到达,看门狗引发Alarm事件
Console.WriteLine("月黑风高的午夜: " + now);
Console.WriteLine("小偷悄悄地摸进了主人的屋内... ");
//自定义参数
UserEventArgs e = new UserEventArgs();
dg.OnAlarm(e);
Console.WriteLine("请按任何键退出~");
Console.ReadKey();
}
} //自定一个事件参数类
class UserEventArgs : EventArgs
{
private int iEventArgs;
public int Args
{
set { iEventArgs = value; }
get { return iEventArgs; }
}
public UserEventArgs(int e)
{
iEventArgs = e;
}
} class Dog
{
//1.声明关于事件的委托;
public delegate void AlarmEventHandler(object sender, UserEventArgs e);
//2.声明事件
public event AlarmEventHandler Alarm;
//3.编写引发事件的函数;
public void OnAlarm(UserEventArgs e)
{
if (Alarm != null)
{
for (int k = ; k < e.Args; k++)
{
Console.WriteLine("汪汪~~");
}
Alarm(this, e);
}
}
}
class Host
{
//主人接收到信息引发的动作
public void Catch(object sender, UserEventArgs e)
{
Console.WriteLine("NND小偷,别跑~");
}
public Host(Dog d)
{
d.Alarm += new Dog.AlarmEventHandler(Catch);
}
}
2.事件订阅的多种写法
在“C#基础之事件(1)”中,用
_A.GetStr += OnGetStr;
来订阅事件,但在上例中用
d.Alarm += new Dog.AlarmEventHandler(Catch);
这种形式来订阅,这两种有什么不同吗?,答案是:没有。前一种是C#2.0的方法,也是我们现在常用的方法;后一种是C#1.0的方法。大致在网络上查找了下,绝大部分示例都是用的后一种写法。在实际项目代码中,用的多的还是前一种写法,毕竟更简洁,看起来也舒服。
从后一种写法可知,其本质是new 了一个委托,并把方法当作委托的参数,参见前面“委托”章节,委托有多种写法,那么当然,除上面两种写法外,事件的订阅就同样有多种简略写法,这时的事件处理逻辑就可以直接写在{}中,而不用单独写成一个方法,实际用哪种方法看个人编码习惯和团队规范要求。现展示如下:
lambda表达式写法
Button1.Click+=(s,e)=>{ }; delegate写法
Button1.Click+=delegate(object sender,EventArgs e) { }; 匿名事件处理
Button1.Click+=delegate { };
3.事件的订阅和取消订阅
事件的订阅用“+=”,取消订阅用“-=”,实际是调用了Add和remove的两个方法,想了解详情请参考《CLR via C#》,这里只从应用角度来说明。需要强调一点:如果是在非构造方法中订阅的事件,在适当的时候需取消订阅事件,如果没有取消当再次调用事件订阅时,会第二次订阅同一事件,这样事件处理方法就会执行第二次,如果执行n 次就会触发n次事件处理方法。
4.多事件返回值的获取
多事件返回值可以通过GetInvocationList方法获取,这个在实际中应用中不是很多,但有应用到。以下从网上找了个示例:
private void button1_Click(object sender, EventArgs e)
{
int Number = ;
Publishser pub = new Publishser();
Subscriber1 sub1 = new Subscriber1();
Subscriber2 sub2 = new Subscriber2();
Subscriber3 sub3=new Subscriber3();
pub.NumberChanged += sub1.OnNumberChanged;
pub.NumberChanged += sub2.OnNumberChanged;
pub.NumberChanged += sub3.OnNumberChanged;
pub.DoComething(Number);
} class Publishser
{
public delegate int DemoEventHandler(int num);
public event DemoEventHandler NumberChanged;
public void DoComething(int temp)
{
if (NumberChanged != null)
{
Delegate[] delArray = NumberChanged.GetInvocationList();
foreach (Delegate del in delArray)
{
DemoEventHandler method = (DemoEventHandler)del;
temp = method(temp);
}
}
MessageBox.Show(temp.ToString());
}
} class Subscriber1
{
public int OnNumberChanged(int num)
{
MessageBox.Show("调用了Subscriber1类,num值为:"+ num);
return num + ; ;
}
} class Subscriber2
{
public int OnNumberChanged(int num)
{
MessageBox.Show("调用了Subscriber2类 num值为:"+num);
return num+;
}
} class Subscriber3
{
public int OnNumberChanged(int num)
{
MessageBox.Show("调用了Subcriber3类,num值为:"+num);
return num+;
}
} 运行得到的结果是: 调用了Subscriber1类,num值为: 调用了Subscriber2类,num值为: 调用了Subscriber3类,num值为:
总结:事件的应用非常广泛,需作为重点来学习。
C#基础之事件(2)的更多相关文章
- [.net 面向对象编程基础] (22) 事件
[.net 面向对象编程基础] (22) 事件 事件(Event)是学习.net面向对象编程很重要的一部分,在学习事件之前,我们实际上已经在很多地方使用了事件,比如控件的click事件等,这些都是. ...
- SQL Server 扩展事件(Extented Events)从入门到进阶(2)——在GUI中创建基础扩展事件
本文属于 SQL Server 扩展事件(Extented Events)从入门到进阶 系列 第一篇文章中提到了如何在Profiler中创建跟踪(trace),并以服务器端(server-side)跟 ...
- jQuery基础之事件
jQuery基础之事件方法,如下图: 代码实现: <script src="JS/jquery-1.12.4.min.js"></script> <s ...
- JS 的事件基础、事件侦听与抛发、
前言 JavaScript是一种事件驱动型语言.事件驱动是指JavaScript引擎并不是在看到代码之后就会立即执行,而是会在合适的时间才去执行.这个合适的时间是指当某个事件发生之后(例如一个输入框的 ...
- JavaScript基础—dom,事件
Js基础-DOM 1:dom:文档对象模型 Dom就是html页面的模型,将每个标签都作为一个对象,js通过调用dom中的属性,方法就可以对网页中的文本框,层等元素进行编程控制.Dom就是一些让jav ...
- 【JS】395-重温基础:事件
本文是 重温基础 系列文章的第二十篇. 这是第三个基础系列的第一篇,欢迎持续关注呀!重温基础 系列的[初级]和[中级]的文章,已经统一整理到我的[Cute-JavaScript](http://js. ...
- NodeJS3-3基础API----event(事件触发器)
1.基础(on) // 如果像对象享有事件能力就要集成EventEmitter const EventEmitter = require('events') //集成EventEmitter类 cla ...
- Delphi - 鼠标上下滚动基础消息事件
Delphi实现对鼠标上下滚动基础消息的截获并处理 前几天有客户提出需求:由于个人PC界面限制,有时候电子图档显示不全,希望通过鼠标上下滚动用来控制电子图档的放大和缩小. 下面通过一个测试Demo来说 ...
- IOS开发UI基础UIControl事件
UIControl事件1.UIControlEventTouchDown单点触摸按下事件:用户点触屏幕,或者又有新手指落下的时候. 2.UIControlEventTouchDownRepeat多点触 ...
随机推荐
- 与karma、angular的第一次亲密接触
首先要了解什么是karma,karma干嘛用的,它的好朋友jasmine又是啥?这些文章可以帮助你: karma干嘛的? angular与karma1 angular与karma2 看了以上几篇文章之 ...
- #华为云·寻找黑马程序员#【代码重构之路】如何“消除”if/else
1. 背景 if/else是高级编程语言中最基础的功能,虽然 if/else 是必须的,但滥用 if/else,特别是各种大量的if/else嵌套,会对代码的可读性.可维护性造成很大伤害,对于阅读代码 ...
- Spring Security OAuth2 Demo —— 隐式授权模式(Implicit)
本文可以转载,但请注明出处https://www.cnblogs.com/hellxz/p/oauth2_impilit_pattern.html 写在前面 在文章OAuth 2.0 概念及授权流程梳 ...
- IOS中获取各个文件的目录路径的方法和NSFileManager类
转自:http://blog.sina.com.cn/s/blog_5fb39f910101di92.html IOS中获取各种文件的目录路径的方法 iphone沙箱模型的有四个文件夹,分别是什么,永 ...
- python笔记02
day02笔记记录 一.今日摘要 循环.字符串格式化.运算符.编码.博客. 二.内容回顾 (一)计算机基础 计算机由硬件和软件组成.传统计算机的硬件一般有输入单元.输出单元,算数逻辑单元.控制单元及记 ...
- JS的引入方式_变量的使用_变量的类型
JS的俩种引入方式: 1. <!--js的引入方式1--> <script> /*网页中的弹框*/ alert("js的学习!!") </script ...
- MDS 多活配置
CephFS 介绍及使用经验分享 阅读 1179 收藏 2 2019-01-14 原文链接:www.jianshu.com WebRTC SFU中发送数据包的丢失反馈juejin.im 目录 Ceph ...
- ARTS-S Why do India and Pakistan keep fighting over Kashmir?
原文 On Wednesday, Pakistani and Indian fighter jets engaged in a skirmish over Indian-controlled terr ...
- 【Web技术】400- 浅谈Shadow DOM
编者按:本文作者:刘观宇,360 奇舞团高级前端工程师.技术经理,W3C CSS 工作组成员. 为什么会有Shadow DOM 你在实际的开发中很可能遇到过这样的需求:实现一个可以拖拽的滑块,以实现范 ...
- Orleans 初接触(二) 测试用例
[返回导航] 在简单了解了Orleans 之后我们可以通过几个例子去加深印象 一.快速入门示例 这个例子也是跟着<Microsoft Orleans 之 入门指南>(https://www ...