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

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

委托(

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

相关推荐