• C#事件详解及应用示例


    简介

            事件是使类具备向其它类通知发生的相关事情的能力。事件被分成两部分:一、引发或发送事件的类(称发布者);二、处理或接收事件的类(称订阅者)。事件也是类型的成员。在 .NET 的桌面应用程序中,按钮、列表框、文本输入框等控件都具有相关的事件,如按键响应、鼠标移动、点击等。

    1、事件模型

            事件一定是发布者(类)的内部逻辑触发的。事件的模型由事件发布者事件成员、事件的响应、事件的处理(回调一个方法)及事件的订阅 个部分组成。事件的订阅是以委托类型为基础的约束事件处理与事件绑定关联一起。下面通过一个图了解事件组成及关系。

    示例:以 C# 中常见的定时器 Timer 为例,了解事件。

    1. using System;
    2. using System.IO;
    3. using System.Timers;
    4. namespace Fountain.Consoles.App
    5. {
    6. internal class Program
    7. {
    8. static void Main(string[] args)
    9. {
    10. try
    11. {
    12. // 事件的发布者
    13. Timer timer = new Timer();
    14. // 触发事件的逻辑
    15. timer.Interval = 2000;
    16. // 事件的订阅者
    17. Log log= new Log();
    18. log.RetentionTime = 20;
    19. // 事件的订阅 (事件处理与事件关联在一起)
    20. timer.Elapsed += log.Delete;
    21. // 事件的发布者触发计时开始
    22. timer.Start();
    23. Console.ReadLine();
    24. }
    25. catch (Exception exception)
    26. {
    27. Console.WriteLine(exception.Message);
    28. }
    29. }
    30. }
    31. ///
    32. /// 日志类
    33. ///
    34. public class Log
    35. {
    36. private object locker = new object();
    37. private int retentionTime = 10;
    38. ///
    39. /// 保留时间
    40. ///
    41. public int RetentionTime
    42. {
    43. get { return retentionTime; }
    44. set { retentionTime = value; }
    45. }
    46. ///
    47. /// 事件处理(删除日志文件)
    48. ///
    49. public void Delete(object sender, ElapsedEventArgs e)
    50. {
    51. // 日志目录
    52. string logDirectory = string.Format("{0}{1}", AppDomain.CurrentDomain.BaseDirectory, "Log");
    53. try
    54. {
    55. lock (this.locker)
    56. {
    57. if(Directory.Exists(logDirectory))
    58. {
    59. // 获取所有匹配的日志文件
    60. string[] files = Directory.GetFiles(logDirectory, "*.log");
    61. // 遍历删除
    62. foreach (string file in files)
    63. {
    64. if (File.GetCreationTime(file).AddDays(this.RetentionTime) < DateTime.Now)
    65. {
    66. File.Delete(file);
    67. Console.WriteLine($"日志文件【{0}】被删除。", file);
    68. }
    69. }
    70. }
    71. }
    72. }
    73. catch (Exception exception)
    74. {
    75. Console.WriteLine($"删除日志文件异常: {0}", exception.Message);
    76. }
    77. }
    78. }
    79. }
    2、事件声明

            .NET 提供了很多带事件的类,如Timer、TextBox、Button 等,对于.NET 提供的,我们只用订阅相关事件,添加事件响应处理方法即可。

            如需自定义一个事件,首先必须在需添加的类中声明该事件的委托类型,然后使用 event 关键字声明事件。

             事件的声明有完整声明简略声明两种方式。下面是事件声明的简单示例:

    1. //声明一个委托类型,专门用来声明事件,约束事件处理器
    2. public delegate void 委托类型名称(处理事件参数);
    3. // 基于委托定义事件
    4. public event 委托类型名称 事件名称;
    5. // 声明委托类型字段
    6. public delegate void TextChangedEventHandler(object sender,KeyEventArgs e);
    7. // 简略声明事件
    8. public static event TextChangedEventHandler TextChangedEvent;
    9. //完整声明
    10. public event TextChangedEventHandler TextChangedEvent
    11. {
    12. add
    13. {
    14. //事件处理器的添加器
    15. this.TextChangedEventHandler += value;
    16. }
    17. remove
    18. {
    19. //事件处理器的移除器
    20. this.TextChangedEventHandler -= value;
    21. }
    22. }
    3、事件属性

    1、发布者确定引发事件时间,订阅者确定对事件如何响应。

    2、事件可被多个订阅,订阅者可以处理多个发布者的事件。

    3、事件未被订阅,则事件不会被引发。

    4、事件一般用于用户操作。

    5、事件被多个订阅时,引发事件时同步调用事件处理程序。

    6、事件基于 EventHandler 与 EventArgs 类。

    4、事件与委托

            事件并不是委托,只是借用委托类型来存储或引用事件处理,基于委托约束事件能够发送什么样的消息给事件的订阅者。把事件作为一种特殊的委托的实例,或者说是受限制的委托,是委托一种特殊应用,只是因为:1、事件声明用到了委托类型;2、可以直接用委托调用事件订阅者处理内容。

    应用

            本应用示例提供应用程序二个界面数据的同步,扫描界面扫描快递单号后,同步到一个放大展示界面。

    事件的参数

    1. using System;
    2. namespace Fountain.WinForm.SyncDisplay
    3. {
    4. ///
    5. /// 事件参数
    6. ///
    7. public class SyncEventArg : EventArgs
    8. {
    9. ///
    10. /// 快递单号
    11. ///
    12. public string TrackingNo { get; set; }
    13. }
    14. }

    事件发布者类,扫描界面:

    1. using System;
    2. using System.Windows.Forms;
    3. namespace Fountain.WinForm.SyncDisplay
    4. {
    5. public partial class ScanForm : Form
    6. {
    7. //使用默认的事件处理委托,定义消息发布者事件
    8. public event EventHandler SendMsgEvent;
    9. ///
    10. ///
    11. ///
    12. public ScanForm()
    13. {
    14. InitializeComponent();
    15. }
    16. ///
    17. /// 界面加载
    18. ///
    19. ///
    20. ///
    21. private void ScanForm_Load(object sender, EventArgs e)
    22. {
    23. try
    24. {
    25. ShowForm showForm = new ShowForm();
    26. // 事件的订阅 (事件处理与事件关联在一起)
    27. SendMsgEvent += showForm.SysncTextChaned;
    28. showForm.Show();
    29. }
    30. catch
    31. {
    32. }
    33. }
    34. ///
    35. /// 文本输入框回车事件
    36. ///
    37. ///
    38. ///
    39. private void TextBoxTrackingNo_KeyDown(object sender, KeyEventArgs e)
    40. {
    41. try
    42. {
    43. if (e.KeyCode == Keys.Enter)
    44. {
    45. if (!string.IsNullOrEmpty(this.TextBoxTrackingNo.Text))
    46. {
    47. // 事件触发通知
    48. SendMsgEvent(this, new SyncEventArg() { TrackingNo = this.TextBoxTrackingNo.Text });
    49. }
    50. }
    51. }
    52. catch
    53. {
    54. }
    55. }
    56. }
    57. }

    事件订阅者类,显示界面:

    1. using System;
    2. using System.Windows.Forms;
    3. namespace Fountain.WinForm.SyncDisplay
    4. {
    5. public partial class ShowForm : Form
    6. {
    7. ///
    8. ///
    9. ///
    10. public ShowForm()
    11. {
    12. InitializeComponent();
    13. }
    14. ///
    15. /// 响应事件处理
    16. ///
    17. ///
    18. ///
    19. internal void SysncTextChaned(object sender, EventArgs e)
    20. {
    21. try
    22. {
    23. //取到主窗体的传来的文本
    24. SyncEventArg arg = e as SyncEventArg;
    25. this.LabelTrackingNo.Text = arg.TrackingNo;
    26. }
    27. catch
    28. {
    29. }
    30. }
    31. }
    32. }

    效果:

    小结

            以上都是事件的相关内容,事件属于C#的高级语法,应用的场景很多。希望通过本篇对大家有帮助,敬请关注后续内容。如需应用示例完整代码关注留言。

  • 相关阅读:
    【Learning eBPF-2】eBPF 的“Hello world”
    阿里云幻兽帕鲁Linux 服务器下载游戏存档的方法
    热点事件情境下微博舆情反转预测
    mycat 常用分片规则使用详解
    BRAM/URAM资源介绍
    算法入门(四):二分法的详解与扩展
    软件测试一些基本面试问题
    程序员的中秋
    Objects.equals有坑
    07-Linux基本权限
  • 原文地址:https://blog.csdn.net/Funniyuan/article/details/139889402