c# 委托 delegate 是什么 有什么用

委托是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 表达式树,底层都靠它撑着。