C#怎么使用委托和事件 C# Delegate和Event入门教程

委托是类型安全的函数指针,用于传递方法;事件是受保护的委托封装,仅允许外部订阅/取消订阅,禁止直接调用或赋值。

委托(Delegate)和事件(Event)是C#中实现回调、松耦合通信的核心机制。简单说:委托是“能装方法的变量”,事件是“受保护的委托,只能被触发、不能被外部直接调用”。掌握它们,才能写好UI响应、异步通知、插件式架构等常见场景。

一、委托:把方法当参数传

委托本质是一个类型安全的函数指针。先定义委托类型,再创建委托实例,最后通过Invoke或直接调用执行绑定的方法。

示例:定义一个处理日志的委托,并用它传递不同日志逻辑:

// 1. 定义委托:返回void,接受一个string参数
public delegate void LogHandler(string message);

// 2. 定义两个符合签名的方法
void WriteToConsole(string msg) => Console.WriteLine($"[CONSOLE] {msg}");
void WriteToFile(string msg) => File.AppendAllText("log.txt", $"[FILE] {msg}\n");

// 3. 创建委托实例并调用
LogHandler logger = WriteToConsole;
logger("程序启动了"); // 输出到控制台

logger = WriteToFile;
logger("用户登录成功"); // 写入文件

✅ 小贴士:

  • 委托类型名推荐用 HandlerCallbackEventHandler 结尾(如 ActionFunc 是内置泛型委托,日常优先用它们)
  • 支持多播:用 += 可绑定多个方法,调用时依次执行(顺序按添加顺序)
  • GetInvocationList() 可查看所有绑定的方法

二、事件:委托的安全封装

事件是基于委托的语法糖,它限制了外部代码只能“订阅(+=)”或“取消订阅(-=)”,不能直接赋值或调用 —— 这保证了发布者对触发时机的完全控制。

典型场景:按钮被点击、文件下载完成、数据验证失败……这些“发生了什么”,由类内部决定何时通知,外部只负责响应。

public class Downloader
{
    // 1. 声明事件(基于内置 EventHandler)
    public event EventHandler DownloadCompleted;

    // 2. 触发事件(内部调用)
    protected virtual void OnDownloadCompleted(DownloadEventArgs e)
    {
        DownloadCompleted?.Invoke(this, e); // 空安全调用
    }

    public void Start()
    {
        // 模拟下载结束
        Thread.Sleep(1000);
        OnDownloadCompleted(new DownloadEventArgs { FileName = "report.pdf", Size = 2048 });
    }
}

public class DownloadEventArgs : EventArgs
{
    public string FileName { get; set; }
    public int Size { get; set; }
}

✅ 订阅方式(在其他类中):

var dl = new Downloader();
dl.DownloadCompleted += (sender, e) =>
{
    Console.WriteLine($"下载完成:{e.FileName} ({e.Size} 字节)");
};
dl.Start();

⚠️ 注意:事件不能在外部写成 dl.DownloadCompleted = ...,编译会报错 —— 这正是它的保护意义。

三、常用模式与避坑点

实际开发中,别从零手写委托类型,优先使用 .NET 提供的泛型委托;事件命名统一用动词过去式(如 ClickedSavedChanged);记得解订阅防内存泄漏(尤其在长期存活对象中监听短命对象事件时)。

  • 用 Func 替代自定义判断委托list.Find(x => x > 10) 底层就是 Func
  • 事件触发前判空:永远用 MyEvent?.Invoke(...),或手动复制一份再调用(线程安全写法):var handler = MyEvent; handler?.Invoke(...)
  • WinForms/WPF 中事件已预置好:比如 button1.Click += ...,背后就是 EventHandler 委托,不用自己定义
  • 避免在事件处理器里做耗时操作:可开 Task 或调用 BeginInvoke 防卡 UI

四、委托 vs 事件 vs Action/Func

一句话区分:

  • 委托类型(如 public delegate void Notify(string s);)是“模板”,描述方法签名
  • Action / Func 是 .NET 预定义的通用委托类型,覆盖绝大多数无返回/有返回场景,够用就别重复造轮子
  • 事件 是委托字段的封装,加了访问限制(只能 += / -=),语义上表示“我这里发生了某事,请你响应”

基本上就这些。不复杂但容易忽略细节 —— 关键是理解“谁控制调用权”:委托让你自由调用,事件只让发布者触发,而 Action/Func 是帮你省去定义委托类型的快捷方式。