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

委托(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("用户登录成功"); // 写入文件
✅ 小贴士:
- 委托类型名推荐用 Handler、Callback 或 EventHandler 结尾(如
Action、Func是内置泛型委托,日常优先用它们) - 支持多播:用
+=可绑定多个方法,调用时依次执行(顺序按添加顺序) - 用
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 提供的泛型委托;事件命名统一用动词过去式(如 Clicked、Saved、Changed);记得解订阅防内存泄漏(尤其在长期存活对象中监听短命对象事件时)。
-
用 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 是帮你省去定义委托类型的快捷方式。









