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

来源:这里教程网 时间:2026-02-21 17:35:25 作者:

委托(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 结尾(如
Action<t></t>
Func<t tresult></t>
是内置泛型委托,日常优先用它们)
支持多播:用
+=
可绑定多个方法,调用时依次执行(顺序按添加顺序)
GetInvocationList()
可查看所有绑定的方法

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

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

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

public class Downloader
{
    // 1. 声明事件(基于内置 EventHandler<T>)
    public event EventHandler<DownloadEventArgs> 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 是帮你省去定义委托类型的快捷方式。

相关推荐