当前位置: 首页 > news >正文

C# 事件(Event)核心概念

文章目录

  • 前言
  • ‌1. 事件的核心概念‌
  • ‌2. 事件的声明与使用‌
  • ‌3. 标准事件模式(EventHandler 和 EventArgs)‌
  • 4. 事件与委托的区别‌
  • ‌5. 事件的使用场景‌
  • ‌6. 高级特性‌
  • ‌7. 注意事项‌
  • ‌8. 完整示例:温度监控系统‌
  • ‌9. 常见问题‌

前言

在 C# 中,‌事件(Event)‌ 是基于委托(Delegate)的机制,用于实现‌发布-订阅(Publish-Subscribe)模式‌。事件允许对象(发布者)通知其他对象(订阅者)某个特定动作已发生(如按钮点击、数据更新)。以下是事件的详细讲解:


‌1. 事件的核心概念‌

  • 发布者(Publisher)‌:触发事件的对象(如按钮控件)。
  • ‌订阅者(Subscriber)‌:响应事件的对象(如事件处理方法)。
  • ‌事件委托(Event Delegate)‌:定义事件的签名(参数和返回值)。
  • ‌封装性‌:事件对外部仅暴露 +=(订阅)和 -=(取消订阅),无法直接触发或覆盖。

‌2. 事件的声明与使用‌

‌(1) 声明事件‌

public class Button
{
    // 1. 定义委托类型(约定返回 void,参数为 object 和 EventArgs 派生类)
    public delegate void ClickEventHandler(object sender, EventArgs e);

    // 2. 声明事件(基于委托)
    public event ClickEventHandler? Clicked;

    // 3. 触发事件的方法(protected virtual 以便派生类重写)
    protected virtual void OnClicked(EventArgs e)
    {
        Clicked?.Invoke(this, e);  // 安全调用(若没有订阅者为 null)
    }

    // 4. 外部触发的入口(如用户点击按钮)
    public void Click()
    {
        OnClicked(EventArgs.Empty);  // 触发事件
    }
}

‌(2) 订阅事件‌

public class Program
{
    public static void Main()
    {
        Button button = new Button();

        // 订阅事件(通过 +=)
        button.Clicked += Button_Clicked;
        button.Clicked += (sender, e) => Console.WriteLine("Lambda 表达式处理点击事件");

        // 取消订阅(通过 -=)
        // button.Clicked -= Button_Clicked;
    }

    // 事件处理方法
    private static void Button_Clicked(object sender, EventArgs e)
    {
        Console.WriteLine("按钮被点击!");
    }
}

‌3. 标准事件模式(EventHandler 和 EventArgs)‌

.NET 提供了标准委托类型 EventHandler 和基类 EventArgs,避免重复定义委托:

// 标准事件声明
public event EventHandler? Clicked;  // 等价于 EventHandler<EventArgs>

// 自定义事件参数
public class TemperatureChangedEventArgs : EventArgs
{
    public double OldTemperature { get; }
    public double NewTemperature { get; }

    public TemperatureChangedEventArgs(double oldTemp, double newTemp)
    {
        OldTemperature = oldTemp;
        NewTemperature = newTemp;
    }
}

// 使用泛型 EventHandler<T>
public event EventHandler<TemperatureChangedEventArgs>? TemperatureChanged;

4. 事件与委托的区别‌

‌特性‌委托(Delegate)‌事件(Event)
访问权限‌可被外部直接调用或赋值‌仅允许 += 和 -= 操作
封装性‌低(暴露委托实例)‌ 高(隐藏触发逻辑)
用途‌ 通用回调机制‌特定于发布-订阅场景

‌5. 事件的使用场景‌

  • UI 交互‌:按钮点击、文本框输入、窗口关闭。
  • ‌数据通知‌:属性值变化、定时器触发。
  • ‌异步操作‌:文件下载完成、网络请求响应。
  • ‌插件系统‌:动态加载模块的事件响应。

‌6. 高级特性‌

‌(1) 自定义事件访问器(add/remove)‌
可自定义事件的订阅和取消订阅逻辑:

private EventHandler? _clicked;

public event EventHandler Clicked
{
    add 
    {
        _clicked += value;
        Console.WriteLine("事件被订阅");
    }
    remove 
    {
        _clicked -= value;
        Console.WriteLine("事件被取消订阅");
    }
}

‌(2) 线程安全的事件触发‌
在多线程场景中,需确保事件触发安全:

// 复制委托引用,避免触发时订阅者被修改
EventHandler? handler = TemperatureChanged;
if (handler != null)
{
    handler(this, e);  // 直接调用,非空时触发
}

‌7. 注意事项‌

  • 内存泄漏‌:若订阅者是对象方法,需及时取消订阅(否则对象无法被 GC 回收)。
  • ‌空事件检查‌:触发事件前检查是否为 null(无订阅者)。
  • 事件命名‌:事件名通常以动词过去式命名(如 Clicked、DataLoaded)。
  • 避免长时间阻塞‌:事件处理应快速完成,避免阻塞发布者线程。

‌8. 完整示例:温度监控系统‌

public class TemperatureSensor
{
    private double _currentTemperature;
    public event EventHandler<TemperatureChangedEventArgs>? TemperatureChanged;

    public double CurrentTemperature
    {
        get => _currentTemperature;
        set
        {
            if (_currentTemperature != value)
            {
                var oldTemp = _currentTemperature;
                _currentTemperature = value;
                OnTemperatureChanged(oldTemp, value);
            }
        }
    }

    protected virtual void OnTemperatureChanged(double oldTemp, double newTemp)
    {
        TemperatureChanged?.Invoke(this, new TemperatureChangedEventArgs(oldTemp, newTemp));
    }
}


// 订阅者
public class Display
{
    public void Subscribe(TemperatureSensor sensor)
    {
        sensor.TemperatureChanged += HandleTemperatureChange;
    }

    private void HandleTemperatureChange(object? sender, TemperatureChangedEventArgs e)
    {
        Console.WriteLine($"温度从 {e.OldTemperature}℃ 变更为 {e.NewTemperature}℃");
    }
}

// 使用
var sensor = new TemperatureSensor();
var display = new Display();
display.Subscribe(sensor);

sensor.CurrentTemperature = 25.5;  // 触发事件

‌9. 常见问题‌

1:为什么事件通常定义为 virtual?‌

  • 允许派生类重写事件触发逻辑(如添加日志或验证)。

2:如何传递自定义数据?‌

  • 继承 EventArgs 并定义包含数据的派生类(如 TemperatureChangedEventArgs)。

3:事件和观察者模式的关系?‌

  • 事件是观察者模式在 C# 中的实现,发布者相当于 Subject,订阅者相当于 Observer。

通过事件,C# 提供了一种类型安全、松耦合的方式来实现对象间的通信,是构建响应式应用程序的核心机制。

相关文章:

  • JVM的垃圾回收器都有哪些?
  • 尚硅谷爬虫(解析_xpath的基本使用)笔记
  • 《算法笔记》9.2小节——数据结构专题(2)->二叉树的遍历 问题 A: 复原二叉树(同问题 C: 二叉树遍历)
  • 小程序电子画册制作,用户体验为王!
  • 【多线程】线程不安全问题
  • 每日学习Java之一万个为什么(待补充)
  • Web Component 教程(四):如何优雅的使用 template 模块
  • springboot集成xxl-job
  • 使用 libmodbus 实现 Modbus 通信
  • linux 出现网卡 down 没起来 怎么办 ? 已解决
  • C/C++编程:Openssl使用 Windows安装包32和64位 RSA加密/解密、AES-GCM加密/解密以及ECDSA签名/验证示例
  • C/C++蓝桥杯算法真题打卡(Day8)
  • 虚幻基础:组件组件通信
  • 一次http请求需要经过哪些步骤?
  • 【GPT入门】第26课 掌握langchain LCEL 链式调用的三种方法
  • Qt msvc程序运行
  • Vue3组合式函数(刷新率 useFps)
  • 搞定python之九----常用内置模块
  • linux环境下快速输出电脑的系统/硬件/显卡/网络/已安装软件等信息
  • AT指令集-NBIOT
  • “五一”逃离城市计划:带上帐篷去大自然里充电
  • 原国家有色金属工业局副局长黄春萼逝世,享年86岁
  • AI观察|算力饥渴与泡沫
  • 国家税务总局:“二套转首套”可以享受贷款利息个税专项扣除
  • 美大学建“私人联盟”对抗政府:学校已存在300年,特朗普才上任3个月
  • 读科学发展的壮丽史诗,也读普通人的传奇