c#笔记5 详解事件的内置类型EventHandler、windows事件在winform中的运用

为什么要研究这一问题?

事件和委托可以说是息息相关。 前面先解释了什么是委托,怎么定义一个委托以及怎么使用匿名方法来内联地新建委托。

事实上事件这一机制在c#的程序开发中展很重要的地位,尤其是接触了winform软件开发的同学们应该都知道界面上的各种操作和事件比如点击事件,文本改变事件等等,都是我们建立用户界面,乃至于实现具体功能的重要手段。于是我们有必要研究这一机制在c#中的默认类型。

EventHandler类

这一类型是.net类库中预定义的一个委托类型,建议我们使用该类型作为事件的委托类型。事实上我们的桌面端winform开发中的事件就是此类。

以我最近写的例子来说,添加了点击事件相当于在按钮的点击事件中添加新的引用,也就是说,按钮是发布者类,其含有一个事件(委托)click,我们写的程序中写的button2_Click函数就作为其响应的方法,注意,这里的订阅操作是默认在设计代码中就完成了,如果你不在设计界面修改点击事件,需要自己操作你的方法对点击事件的引用和订阅。同样的,这里的click事件也只能由按钮本身触发,而不能手动在代码写一个click()之类的,事实上它的原型应该需要的参数和我们的EventHandler一样;

图1:按钮的点击方法对点击事件的订阅 

图2:click方法的定义以及两个访问器

图3:事件本质是对委托的封装

 add和remove是特殊的访问器。

图4:与EventHandler类相匹配的方法

图5:EventHandler类的定义

如此可以了解该类与委托和事件的关系。他是经过了访问器修改的委托类型,是定义事件的推荐类型。

EventHandler类如何使用

该类具有两个实例化参数,对于一个委托而言,这意味着其对应引用的方法需要与该模式匹配。

第一个参数object sender

这就是委托的触发者,或者说事件的触发者,在事件机制下,这一对象只能是发布者类自己的对象。比如按钮的点击事件就只能是按钮本身触发。

触发时传给该参数的值通常为this,注意这里的参数并不是我们写方法时要传入的,而是事件触发时传入的。

比如我们的click事件原型来自于control类的:

    protected virtual void OnClick(EventArgs e){((EventHandler)base.Events[EventClick])?.Invoke(this, e);}

button->buttonbase->control

定义来源如上所示。

也就是说这里的sender是触发时,由发布者类或者说触发者提供给我们的信息:发布者本身的引用。我们凭借这一个引用可以获取触发者的各种信息和方法。

在实际使用中我们一般不需要特别修改这个逻辑,或者说我们如果自定义一个事件,也要类似这样定义这一参数。

第二个参数EventArgs e

这一参数就是真正触发事件时用于传参的结构了,虽然只是一个参数,但是通过建立新的类来继承EventArgs,之后再实例化我们自己的参数类,就可以用于传递我们想要的参数结构了,但是需要注意的是:这不意味着我们必须要传参,这个类内置empty字段,允许我们使用此字段作为参数传递。

public class MyButton
{public event EventHandler Click;public void OnClick(){// 触发 Click 事件,不传递任何数据Click?.Invoke(this, EventArgs.Empty);}
}

传参的例子:

// TimerEventArgs.cs
public class TimerEventArgs : EventArgs
{public int ElapsedSeconds { get; private set; }public TimerEventArgs(int elapsedSeconds){ElapsedSeconds = elapsedSeconds;}
}// Timer.cs
public class Timer
{public event EventHandler<TimerEventArgs> SecondElapsed;private int seconds;public Timer(){// 假设这里有一些逻辑来启动计时器}// 这个方法模拟计时器每秒触发一次事件public void Tick(){seconds++;OnSecondElapsed(seconds);}protected virtual void OnSecondElapsed(int elapsedSeconds){SecondElapsed?.Invoke(this, new TimerEventArgs(elapsedSeconds));}
}

 可以看到我们定义了一个 TimerEventArgs类来传递参数,包含一个只读字段和构造函数,在触发者类中给出初始化参数的方法以及触发我们的SecondElapsed事件时进行传递,在下文的主函数中,我们实例化发布者类,然后订阅了我们定义的private static void Timer_SecondElapsed函数(这一操作也可以说是我们的方法订阅了这个事件),或者说我们的方法添加到了委托中。之后就是让我们的发布者类触发十次该事件。

public class Program
{public static void Main(){Timer timer = new Timer();timer.SecondElapsed += Timer_SecondElapsed;// 模拟计时器运行for (int i = 0; i < 10; i++){timer.Tick();Thread.Sleep(1000); // 等待一秒钟}}private static void Timer_SecondElapsed(object sender, TimerEventArgs e){Console.WriteLine($"Timer elapsed {e.ElapsedSeconds} seconds.");}
}

可以看到,在我们的事件响应方法中使用到了我们传入的参数,并读取了那个代表已经过去时间的只读字段。

注意咯:如果我们在事件处理中不是定义了这样:

 private static void Timer_SecondElapsed(object sender, TimerEventArgs e)

而是这样:

 private static void Timer_SecondElapsed(object sender, EventArgs e)

那么传入参数之后还需要进行强转哦:

private static void Timer_SecondElapsed(object sender, EventArgs e)
{Console.WriteLine($"Timer elapsed {(TimerEventArgs)e.ElapsedSeconds} seconds.");
}

以我粗浅的见识来看,两个方法都可以。

Windows事件

这一话题我曾经非常感兴趣,这里的Windows事件与广义的操作系统中的机制不同,我们谈论的是我们触发事件的条件与我们在Windows操作系统中的操作有关时,我们就可以理解为我们这里说的Windows事件。

比如我们在图形界面中点击了一个按钮,我们都知道这样一个点击事件会被触发,但是怎么触发的呢,一定有什么东西接收了鼠标的操作,最后由什么东西传递给了我们程序,然后程序调用了这个委托事件,然后我们写的代码才得以执行。

在Windows Forms应用程序中,按钮(Button)控件的Click事件最初是由Windows操作系统通过消息队列触发的。当用户点击按钮时,操作系统会生成一个消息,并将其放入应用程序的消息队列中。这个消息通常具有特定的消息代码(如WM_LBUTTONDOWN、WM_LBUTTONUP等),表示鼠标按钮被按下和释放。

Windows Forms控件的WndProc方法负责处理这些消息。WndProc是控件基类(通常是Control类)的一个方法,它负责处理发送到控件的所有Windows消息。当控件接收到与按钮点击相关的消息时,它会根据消息的内容执行相应的操作,并可能触发相应的Click事件。

 

图6:button的触发事件

下面为接收操作系统消息之后触发事件的代码:

 protected override void WndProc(ref Message m){int msg = m.Msg;if (msg == 245){if (this is IButtonControl){((IButtonControl)this).PerformClick();}else{OnClick(EventArgs.Empty);}return;}if (OwnerDraw){switch (m.Msg){case 8:case 31:case 533:if (!GetFlag(8) && GetFlag(4)){SetFlag(4, value: false);if (GetFlag(2)){SetFlag(2, value: false);Invalidate(DownChangeRectangle);}}base.WndProc(ref m);break;case 514:case 517:case 520:try{SetFlag(8, value: true);base.WndProc(ref m);break;}finally{SetFlag(8, value: false);}default:base.WndProc(ref m);break;case 243:break;}return;}int msg2 = m.Msg;if (msg2 == 8465){if (NativeMethods.Util.HIWORD(m.WParam) == 0 && !base.ValidationCancelled){OnClick(EventArgs.Empty);}}else{base.WndProc(ref m);}}

 最后能够溯源到control类的同名函数:

具体的逻辑需要一层层剖析,但是不可否认,这就是我们能够通过外设操作计算机程序的关键。 

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

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

相关文章

Unity 动态光照贴图,加载后显示变暗或者变白问题 ReflectionProbe的使用

动态加载光照贴图代码&#xff0c;可参考这个帖子 Unity 预制动态绑定光照贴图遇到变白问题_unity urp 动态加载光照信息 变黑-CSDN博客 这次遇到的问题是&#xff0c;在编辑器下光照贴图能正常显示&#xff0c;打出apk后光照贴图加载后变黑的问题 以下4张图代表4种状态&…

duckDB源码GDB调试

duckDB源码GDB调试 一&#xff0c;首先编译源代码&#xff0c;使用debug模式 在根目录下 make debug cd ./build/debug/test二&#xff0c;gdb调试unittest gdb ./unittest三&#xff0c;打断点 b dictionary_compression.cpp:621四&#xff0c;执行unittest run test/sq…

opencascade 重叠曲线设置优先显示

‌OpenCASCADE重叠曲线显示设置‌ 当出现重叠曲线时&#xff0c;往往需要设置 优先显示的对象 关键点 SetDisplayPriority SetLayer

磁性齿轮箱市场报告:前三大厂商占有大约79.0%的市场份额

磁性齿轮箱是一种用于扭矩和速度转换的非接触式机构。它们无磨损、无摩擦、无疲劳。它们不需要润滑剂&#xff0c;并且可以针对其他机械特性&#xff08;如刚度或阻尼&#xff09;进行定制。 一、全球磁性齿轮箱行业现状与洞察 据 QYResearch 调研团队最新发布的“全球磁性齿轮…

在 Next.js 项目中使用 Clerk 实现 OAuth 登录与回调处理

1. 什么是 Clerk 和 OAuth 登录&#xff1f; Clerk 是一个身份验证管理工具&#xff0c;可以轻松集成到 Web 应用中&#xff0c;提供安全的用户登录、注册等功能。OAuth 是一种常见的授权协议&#xff0c;允许用户通过第三方服务&#xff08;如 Google、Microsoft&#xff09;…

10分钟了解OPPO中间件容器化实践

背景 OPPO是一家全球化的科技公司&#xff0c;随着公司的快速发展&#xff0c;业务方向越来越多&#xff0c;对中间件的依赖也越来越紧密&#xff0c;中间件的集群的数量成倍数增长&#xff0c;在中间件的部署&#xff0c;使用&#xff0c;以及运维出现各种问题。 1.中间件与业…

PCM转PCMA(pcm_alaw,G711.A率)转换表 PCM转PCMU(pcm_ulaw,G711.U率)转换表

PCM转PCMA&#xff08;pcm_alaw&#xff0c;G711.A率&#xff09;转换表 && PCM转PCMU&#xff08;pcm_ulaw&#xff0c;G711.U率&#xff09;转换表 文章目录 PCM转PCMA&#xff08;pcm_alaw&#xff0c;G711.A率&#xff09;转换表 && PCM转PCMU&#xff08;…

桥接与NET

仔细看看下面两幅图 net模式&#xff0c;就是在你的Windows电脑&#xff08;假设叫A电脑&#xff09;的网络基础上&#xff0c;再生成一个子网络&#xff0c;ip的前两位默认就是192.168&#xff0c;然后第三位是随机&#xff0c;第四位是自己可以手动设置的。使用这种模式唯一的…

Centos 7 升级glibc2.33 记录

查看glibc升级所需的依赖编译环境&#xff1a; cd $HOME/glibc-$glibc_ver cat INSTALL | grep -E "newer|later" glibc升级参考链接&#xff1a; 主要参考链接&#xff1a;https://blog.csdn.net/optimistic001/article/details/136705055 次要参考链接&#x…

设计模式结构型模式之代理模式

结构型模式之代理模式 一、概念和使用场景1、概念2、核心思想3、java实现代理模式的方式4、使用场景 二、示例讲解1. 静态代理2. 动态代理 三、总结1、使用规则2、代理模式的优点包括&#xff1a;3、代理模式的缺点包括&#xff1a; 一、概念和使用场景 1、概念 代理模式是一…

如何设计数据库排序字段

最简单的办法就是按照id进行排序&#xff0c;越小的id排序越前&#xff0c;不过这完全没有灵活性可言&#xff0c;所以 int字段作为排序 采用一个额外的int字段作为排序成为更普遍的方式 考虑现实场景中&#xff0c;很多时候是需要进行中间插入排序的&#xff0c;这意味要求…

Numpy 数组及矩阵创建详解

基本数组创建方式 numpy中的主要数据类型为ndarray类型,也可以称之为数组,其在内存中是连续存储的.numpy底层大多C语言进行编写,所以运行效率较高,并且numpy库支持并行计算,如矩阵乘法以及其他线性代数操作 np.array() np.array()是numpy中最为基础和常用的创建数组的方式,其…

HUD杂散光环境模拟测试设备

概述 HUD&#xff08;Head-Up Display&#xff09;杂散光环境模拟测试设备是用于模拟飞行器在实际运行过程中可能遇到的多种光照环境的系统。它主要用于测试和验证HUD显示系统的性能&#xff0c;确保其能在各种光线条件下清晰、准确地显示信息&#xff0c;从而保障飞行员在复杂…

redis面试(二十六)总结

到这里 redis的相关知识就要告一段落了&#xff0c;给前面的系列文章做一下大致的总结。 前四篇是redis底层数据结构实现逻辑剖析&#xff0c;四、五两篇说的是持久化和淘汰策略&#xff0c;后面大量篇幅讲的都是redis锁相关。 其他相关的一些面试问题&#xff0c;之前也发过…

深入了解Cassandra数据库:原理、架构与最佳实践

一、Cassandra的基本原理与架构 1.1 分布式架构 Cassandra的架构是无中心化的&#xff0c;这意味着每个节点在集群中都是平等的&#xff0c;没有单一的主节点。这种设计确保了系统的高可用性&#xff0c;即使在部分节点失效的情况下&#xff0c;集群依然可以正常运行。Cassan…

MySQL——多表操作(四)(2)带 EXISTS 关键字的子查询

EXISTS 关键字后面的参数可以是任意一个子查询&#xff0c;这个子查询的作用相当于测试&#xff0c;坏产生任何数据&#xff0c;只返回 TRUE 或 FALSE&#xff0c;当返回值为 TRUE 时&#xff0c;外层查询才会执行。 例如&#xff0c;查询 employee 表中是否存在年龄大于 21岁的…

集成电路学习:什么是BLE低功耗蓝牙

一、BLE&#xff1a;低功耗蓝牙 BLE&#xff0c;即低功耗蓝牙&#xff08;Bluetooth Low Energy&#xff09;&#xff0c;也被称为蓝牙4.0&#xff0c;是蓝牙技术的一种变体。BLE的主要特点在于其低功耗特性&#xff0c;旨在通过一系列的技术和优化措施&#xff0c;使得设备能够…

【大模型理论篇】通用大模型架构分类及技术统一化

1. 背景 国内的 “百模大战” 以及开源大模型的各类评测榜单令人眼花缭乱&#xff0c;极易让人陷入迷茫。面对如此众多的大模型&#xff0c;我们该如何审视和选择呢&#xff1f;本文将从大模型架构的角度&#xff0c;对常见的开源大模型架构进行汇总与分析。资料来源于公开…

MySQL——多表操作(三)连接查询(1)交叉连接

在关系型数据库管理系统中&#xff0c;建立表时各个数据之间的关系不必确定&#xff0c;通常将每实体的所有信息存放在一个表中&#xff0c;当查询数据时&#xff0c;通过连接操作查询多个表中的实体信息&#xff0c;当两个或多个表中存在相同意义的字段时&#xff0c;便可以通…

2024全国大学生数学建模国赛,成员如何分工协作?

文末获取2024国赛数学建模思路代码&#xff0c;9.5开赛后第一时间更新 大家知道&#xff0c;数学建模竞赛是需要一个团队的三个人在三天或四天的时间内&#xff0c;完成模型建立&#xff0c;编程实现和论文写作的任务&#xff0c;对许多第一次参加建模或者建模经验比较欠缺的团…