新闻中心

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

2025-12-08
浏览次数:
返回列表
委托链通过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配音

在线文字转语音软件-专业的配音网站

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

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

调用时如何执行整个链?

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

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

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

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

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


# 第一个  # 荔枝网络营销的推广方案  # 网络推广网站选哪家公司  # 秦皇岛企业网络推广营销  # 资溪电商网站建设平台  # 江都网站优化推广  # 宜昌网站排名优化  # 公证网站建设建议  # 嘉祥营销推广软件  # 宁德新媒体营销推广  # 关键词排名覀金手指花总  # 实际应用  # 回调  # 更有  # 委托  # 数据结构  # 子类  # 运算符  # 链式  # 遍历  # 移除  # gate.  # gate  # .net  # 区别  # ai  # c# 


相关栏目: 【 科技资讯46185 】 【 网络学院92790


相关推荐: uc手机浏览器网页版入口 uc浏览器手机版便捷登录首页  Composer如何处理Git子模块(submodule)依赖_Composer与Git Submodule的对比与选择  sublime怎么预览Markdown渲染效果_Markdown Preview插件 for sublime教程  Go语言中动态执行代码字符串的策略与实践  消息称三星明年 2 月正式发布 HBM4,与 SK 海力士同台竞技  sublime如何配置Python开发环境_将sublime打造成轻量级Python IDE  创客贴用户入口官网登录 创客贴网页版电脑版系统  HTML空白字符处理机制:渲染、DOM与编码实践  抖音创作助手登录入口_抖音创作辅助工具官网直达  苹果手机指南针不准怎么校准 传感器校准方法详解【建议收藏】  优化MinIO list_objects_v2 操作的性能瓶颈与最佳实践  必由学官网快捷入口 必由学网页版在线学习平台  邮政编码查询不到怎么办_邮政编码查询不到的常见原因与对策  J*aScript井字棋(Tic-Tac-Toe)核心交互逻辑实现教程  优化Django表单:提交验证失败后保留用户输入  一加手机电池耗电快怎么办_一加手机电池耗电快的解决方法  必由学网页版入口 必由学官方平台直接访问  C++ vector二维数组定义_C++ vector of vector用法  vivo手机参数配置怎么增强信号_vivo手机参数配置信号增强方法  谷歌浏览器一键优化方案_谷歌浏览器直达主页极速不卡版  J*aScript中向JSON对象添加新属性的正确姿势  J*aScript教程:根据元素文本内容动态设置背景色  Golang如何测试channel通信行为_Golang channel通信测试与分析方法  深入理解J*aScript Promise异步执行与微任务队列  神经网络二分类模型训练异常:高损失与完美验证准确率的排查与修正  Safari自带网页翻译功能怎么用 无需插件轻松看懂外文网站【方法】  qq游戏网页版直接玩_qq游戏免下载快速入口  响应式CSS Grid布局:优化网格项在小屏幕下的堆叠与宽度适配  修复二维数组索引越界异常:一维循环到二维坐标的正确映射  Win11输入法不见了怎么办_Windows11恢复语言栏显示方法  2026年发布! 美少女养成动作RPG《神剑少女战记》发布实机演示  cad如何更改注释性对象的比例_cad注释性比例调整方法  TikTok搜索结果不显示如何解决 TikTok搜索刷新优化方法  痛风发作了怎么办? 快速止痛和后期饮食调理  文心一言怎样用插件调度API数据_文心一言用插件调度API数据【API调用】  Python多线程中正确使用sigwait处理SIGALRM信号  Lar*el头像管理:图片缩放与旧文件删除的最佳实践  微博网页版首页入口 微博电脑端官网登录链接  树莓派传感器触发:通过Twilio API发送WhatsApp消息教程  如何在CSS中使用visited与link控制链接颜色_visited link伪类配合  怎样在Excel中做仪表盘_Excel仪表盘设计与关键指标展示方法  NetBeans Ant项目:自动化将资源文件复制到dist目录的教程  QQ邮箱网页版入口登录 QQ邮箱在线邮箱官方通道  如何使 Jest 模拟函数默认抛出错误以提高测试效率  J*aScript中高效清空DOM列表元素:解决for循环中断与任务管理问题  必由学在线入口 必由学网页版快速登录入口  Node.js中HTML按钮与J*aScript函数交互的正确姿势  python3时间如何用calendar输出?  Angular响应式表单:实现提交后表单及按钮的禁用与只读化  Pygame教程:解决用户输入与游戏状态更新不同步问题 

搜索