委托是C#中可实例化的类型安全函数指针,本质为继承MulticastDelegate的密封类,支持多播、反射、序列化和泛型;它使“行为”成为一等公民,支撑事件、异步、LINQ等核心机制。

委托(delegate)在 C# 中不是“语法糖”或“高级技巧”,它是一个**可实例化的类型安全函数指针**——你可以把它当成一个“方法的容器”,能存、能传、能调、还能链式执行。
委托本质是类,不是语法别名
编译器看到 public delegate int MathOp(int a, int b);,会自动生成一个继承自 System.MulticastDelegate 的密封类。这意味着:
- 它支持
+=和-=,因为底层是多播委托链(GetInvocationList()可查) - 它能被反射识别、能序列化(需标记
[Serializable])、能作为泛型参数(如Action) - 它不是“写法简写”,而是真实存在的类型:你甚至可以
typeof(MathOp).BaseType查到它是MulticastDelegate
为什么非用 delegate 而不是直接传方法名?
因为 C# 不允许把方法名当值直接传递(比如不能写 SomeMethod(Add),除非 Add 是委托类型变量)。委托解决了这个根本限制:
-
解耦回调逻辑:比如
Task.ContinueWith(Action,你传进去的不是某个具体类的方法,而是一个可随时替换的委托实例continuation) -
统一接口,动态绑定:排序时传
Array.Sort(arr, (x,y) => x.CompareTo(y)),背后是Comparison委托;你换 Lambda 就换行为,不用改排序算法本身 -
事件机制的基础:
public event EventHandler中的DataReceived; EventHandler就是委托类型,+=实际是在操作委托链
常见误用:混淆 Action/Func 与自定义 delegate
很多人一上来就手写 public delegate void LogHandler(string msg);,但其实绝大多数场景该用内置泛型委托:
- 无返回值、0~16 个参数 → 用
Action<...>(如Action) - 有返回值、1~17 个参数(最后一个泛型是返回类型)→ 用
Func<...>(如Func) - 只有当你需要**命名语义**(比如强调这是“校验规则”而非普通函数)或**跨程序集公开 API** 时,才定义具名委托
否则,手写委托反而增加维护成本,且和 LINQ、TPL 等现代 API 风格不一致。
最容易踩的坑:null 引用和线程安全
委托变量可能为 null,直接调用会抛 NullReferenceException;多播委托在并发修改(如 UI 线程 +=,后台线程 -=)时可能崩。正确做法是:
- 调用前判空:
if (callback != null) callback("ok");或更简洁地callback?.Invoke("ok"); - 事件发布时标准写法:
var handler = MyEvent; if (handler != null) handler(this, e);(C# 6+ 可简化为MyEvent?.Invoke(this, e);) - 避免在多线程中直接修改同一委托实例;如需线程安全广播,考虑用
ConcurrentDictionary管理回调,或用lock同步委托链操作
委托真正难的不是语法,而是理解它如何让“行为”变成一等公民——你可以存储它、组合它、延迟执行它、跨线程传递它。一旦跳过“它只是个指针”的直觉层,就会发现所有事件、异步延续、策略模式、LINQ 表达式树,底层都靠它撑着。








