委托(
delegate)在 C# 中不是“语法糖”或“高级技巧”,它是一个**可实例化的类型安全函数指针**——你可以把它当成一个“方法的容器”,能存、能传、能调、还能链式执行。
委托本质是类,不是语法别名
编译器看到
public delegate int MathOp(int a, int b);,会自动生成一个继承自
System.MulticastDelegate的密封类。这意味着: 它支持
+=和
-=,因为底层是多播委托链(
GetInvocationList()可查) 它能被反射识别、能序列化(需标记
[Serializable])、能作为泛型参数(如
Action<t></t>) 它不是“写法简写”,而是真实存在的类型:你甚至可以
typeof(MathOp).BaseType查到它是
MulticastDelegate
为什么非用 delegate 而不是直接传方法名?
因为 C# 不允许把方法名当值直接传递(比如不能写
SomeMethod(Add),除非
Add是委托类型变量)。委托解决了这个根本限制: 解耦回调逻辑:比如
Task.ContinueWith(Action<task> continuation)</task>,你传进去的不是某个具体类的方法,而是一个可随时替换的委托实例 统一接口,动态绑定:排序时传
Array.Sort(arr, (x,y) => x.CompareTo(y)),背后是
Comparison<t></t>委托;你换 Lambda 就换行为,不用改排序算法本身 事件机制的基础:
public event EventHandler<dataeventargs> DataReceived;</dataeventargs>中的
EventHandler就是委托类型,
+=实际是在操作委托链
常见误用:混淆 Action/Func 与自定义 delegate
很多人一上来就手写
public delegate void LogHandler(string msg);,但其实绝大多数场景该用内置泛型委托: 无返回值、0~16 个参数 → 用
Action<...></...>(如
Action<string int></string>) 有返回值、1~17 个参数(最后一个泛型是返回类型)→ 用
Func<...></...>(如
Func<int string bool></int>) 只有当你需要**命名语义**(比如强调这是“校验规则”而非普通函数)或**跨程序集公开 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 表达式树,底层都靠它撑着。
