C# 委托链(delegate chaining)的实现原理 - +和-操作符的背后

煙雲
发布: 2025-12-08 22:04:45
原创
266人浏览过
委托链通过Delegate.Combine和Remove实现,基于MulticastDelegate的_invocationList数组合并与移除,+操作符创建新实例合并调用列表,-操作符逆序查找并移除最后一个匹配项,调用时遍历执行各节点,异常会中断后续调用,需手动遍历GetInvocationList实现容错。

c# 委托链(delegate chaining)的实现原理 - +和-操作符的背后

委托链的实现原理,本质上是 Delegate 类对 +- 操作符的重载,底层依赖于 MulticastDelegate 的链式结构和不可变性设计。

委托链的底层数据结构:MulticastDelegate

C# 中所有多播委托(即支持链式调用的委托)都继承自 MulticastDelegate,而它本身又继承自 Delegate。关键区别在于:
- Delegate 表示单个方法调用(Target + Method
- MulticastDelegate 额外维护一个 private readonly object[] _invocationList 字段(.NET Core/.NET 5+ 中为 object?[]),实际存储委托节点数组

这个数组不是动态扩容的列表,而是每次组合时新建数组,体现“不可变性”——这也是委托链线程安全的基础。

+ 操作符:创建新委托链

当写 del1 + del2 时,编译器调用 Delegate.Combine(del1, del2),其逻辑大致如下:

  • del1null,返回 del2(非 null)
  • del2null,返回 del1
  • 若两者都不为 null,且都是 MulticastDelegate 子类,则合并它们的 _invocationList 数组(先拷贝 del1 的全部项,再追加 del2 的全部项)
  • 若任一为单播委托(即普通 Delegate),则将其包装为长度为 1 的数组参与合并

最终返回一个全新的 MulticastDelegate 实例,原委托对象不变。

- 操作符:从链中移除匹配的委托

del1 - del2 对应 Delegate.Remove(del1, del2),行为更精细:

墨狐AI
墨狐AI

5分钟生成万字小说,人人都是小说家!

墨狐AI 249
查看详情 墨狐AI
  • del1 的调用列表尾部开始向前扫描(注意:是逆序)
  • 逐个比对每个委托节点是否与 del2 “相等”(通过 Delegate.Equals() 判定:要求 TargetMethod 完全一致)
  • 找到第一个匹配项后,移除它,并将剩余项组成新数组,返回新委托
  • 若没找到,直接返回原 del1

⚠️ 注意:只移除最后一个匹配项**(不是第一个),且仅移除一个。多次减同一委托需多次调用 -

调用时如何执行整个链?

当你调用 multiDel(),实际触发的是 MulticastDelegate.Invoke() 的默认实现:

  • 遍历 _invocationList 中每个委托
  • 依次调用 eachDelegate.Invoke(...)
  • 如果中间某个委托抛出异常,默认不会自动跳过**,整个调用立即中断(除非你手动捕获)

如需容错执行(类似“fire-and-forget”风格),需自行遍历 GetInvocationList() 并 try-catch 每一项。

基本上就这些。委托链不是魔法,它是基于不可变数组拼接 + 运算符重载 + 继承体系的一套严谨设计,既保证语义清晰,又兼顾线程安全与可预测性。

以上就是C# 委托链(delegate chaining)的实现原理 - +和-操作符的背后的详细内容,更多请关注php中文网其它相关文章!

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号