0

0

如何在Golang中应用中介者模式 Go语言降低多个对象间通信复杂度

P粉602998670

P粉602998670

发布时间:2026-03-08 09:42:11

|

534人浏览过

|

来源于php中文网

原创

go中中介者模式不依赖继承而靠接口和组合实现,核心是“谁持有谁、谁调用谁”;应避免同事类强引用中介者,改用闭包、回调或事件注册被动接入,并区分命令流与事件流合理使用channel。

如何在golang中应用中介者模式 go语言降低多个对象间通信复杂度

中介者模式在 Go 里为什么不用类继承也能实现

Go 没有传统面向对象的继承机制,所以没法靠「抽象中介者 + 具体中介者」那一套 UML 图里的写法。但它的接口(interface{})和组合能力,反而让中介者更轻量——关键不是“谁继承谁”,而是“谁持有谁、谁调用谁”。

常见错误是硬套 Java/C# 写法,比如试图定义一个 Mediator 接口然后让所有同事类都依赖它,结果发现同事之间还是得互相传参、强引用中介者实例,通信路径没真正收敛。

  • 真正该做的:把中介者设计成纯函数式协调器,同事对象只暴露 Notify()HandleEvent() 这类方法,内部逻辑由中介者统一调度
  • 同事对象不保存对中介者的引用,而是通过闭包、回调或事件注册方式“被动接入”
  • 避免在同事结构体字段里直接嵌入 *Mediator —— 这会制造循环依赖,也违背松耦合本意

用 channel 实现松耦合中介者时的典型阻塞问题

很多人一想到“解耦通信”,立刻用 chan 做消息总线,但很快遇到 goroutine 卡死、消息丢失或 panic:send on closed channel

根本原因不是 channel 用错了,而是没区分「命令流」和「事件流」:命令需要响应,事件只需广播;前者适合带缓冲的 channel + 显式应答,后者更适合 select + default 防阻塞。

立即学习go语言免费学习笔记(深入)”;

Clipfly
Clipfly

一站式AI视频生成和编辑平台,提供多种AI视频处理、AI图像处理工具。

下载
  • 别用无缓冲 channel 让同事直接 mediator.events —— 一旦没人接收就永久阻塞
  • 给事件 channel 加缓冲(如 make(chan Event, 16)),并在中介者启动 goroutine 消费,用 select { case 避免积压
  • 如果同事需要确认执行结果,改用 func(Event) error 回调,而不是等待 channel 返回值

当多个模块都要注册到同一中介者时,如何避免重复注册或竞态

比如用户模块、订单模块、通知模块都调用 mediator.Register(UserHandler{}),但没做类型检查或去重,导致同个事件被处理两次,或者 map 并发写 panic。

Go 的运行时不会帮你校验注册唯一性,这事必须手动兜底。

  • 注册表用 sync.Map 替代普通 map,尤其当注册/注销发生在不同 goroutine 时
  • 用接口的指针地址(fmt.Sprintf("%p", handler))或自定义 ID() string 方法做 key,别用反射名(reflect.TypeOf(h).Name())—— 同名不同实例会冲突
  • 提供 Unregister() 并确保它幂等:注销不存在的 handler 不 panic,也不报错

中介者不该承担业务逻辑,但 Go 里容易越界写成“上帝对象”

你写了个 OrderMediator,里面开始处理库存扣减、发 Kafka、更新 Redis 缓存……这已经不是中介者,是业务编排层了。问题在于:测试难、复用差、职责模糊。

中介者的边界很朴素:只转发、只转换、只协调顺序。真正的业务动作,应该交给独立的 service 或 usecase 包。

  • 中介者方法里只出现 userSvc.Create()notify.Send() 这类调用,绝不出现 if stock
  • 参数传递用 DTO(如 type OrderCreatedEvent struct { OrderID string; UserID string }),而不是直接传 *Order 实体——防止同事对象误改共享状态
  • 如果发现中介者文件超过 200 行、import 超过 5 个业务包,大概率该拆了

最难把握的是“协调时机”:什么时候该由中介者触发下一步,什么时候该让上游自己决定。这个分寸没有银弹,只能靠日志埋点 + 一次上线后看调用链路是否清晰。一旦发现 A → B → C 的链路里,B 总是机械转发,那 B 就该消失了。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
golang如何定义变量
golang如何定义变量

golang定义变量的方法:1、声明变量并赋予初始值“var age int =值”;2、声明变量但不赋初始值“var age int”;3、使用短变量声明“age :=值”等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

210

2024.02.23

golang有哪些数据转换方法
golang有哪些数据转换方法

golang数据转换方法:1、类型转换操作符;2、类型断言;3、字符串和数字之间的转换;4、JSON序列化和反序列化;5、使用标准库进行数据转换;6、使用第三方库进行数据转换;7、自定义数据转换函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

246

2024.02.23

golang常用库有哪些
golang常用库有哪些

golang常用库有:1、标准库;2、字符串处理库;3、网络库;4、加密库;5、压缩库;6、xml和json解析库;7、日期和时间库;8、数据库操作库;9、文件操作库;10、图像处理库。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

355

2024.02.23

golang和python的区别是什么
golang和python的区别是什么

golang和python的区别是:1、golang是一种编译型语言,而python是一种解释型语言;2、golang天生支持并发编程,而python对并发与并行的支持相对较弱等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

214

2024.03.05

golang是免费的吗
golang是免费的吗

golang是免费的。golang是google开发的一种静态强类型、编译型、并发型,并具有垃圾回收功能的开源编程语言,采用bsd开源协议。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

407

2024.05.21

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

470

2025.06.09

golang相关判断方法
golang相关判断方法

本专题整合了golang相关判断方法,想了解更详细的相关内容,请阅读下面的文章。

200

2025.06.10

golang数组使用方法
golang数组使用方法

本专题整合了golang数组用法,想了解更多的相关内容,请阅读专题下面的文章。

1377

2025.06.17

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

23

2026.03.06

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go 教程
Go 教程

共32课时 | 5.9万人学习

Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

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