如何使用Golang实现中介者模式_Golang中介者模式对象协作方法

发布时间 - 2026-01-21 00:00:00    点击率:
Go中中介者模式不用class和继承,因Go无类与继承机制;应以结构体承载协调逻辑,用函数字段或方法注入行为,同事仅弱引用中介者,避免循环依赖与过度耦合。

中介者模式在 Go 里为什么不用 class 和继承

Go 没有类、没有继承、也没有 abstract 关键字,所以传统 UML 里的 Mediator 抽象类和 Colleague 基类无法直接翻译。强行用接口模拟抽象类,反而会让协作逻辑散落在各处,失去中介者“集中协调”的本意。

更自然的做法是:用一个结构体承载协调逻辑,用函数字段或方法接收具体行为,同事对象只保留对中介者的弱引用(如 *Mediator 或函数类型)。

  • 同事不持有其他同事的引用,只认得中介者
  • 中介者持有所有需要协调的组件(可以是 struct 字段、map、或切片)
  • 避免循环导入:把中介者定义放在独立包,或与核心业务逻辑同层

如何定义可复用的 Mediator 结构体

中介者不是模板,而是针对具体协作场

景定制的协调器。比如聊天室、订单状态同步、UI 组件联动——每种场景下,“谁通知谁”“什么条件下转发”都不同。

典型结构包含三部分:被协调对象的注册容器、事件分发逻辑、以及可注入的响应函数。

type ChatRoom struct {
	users map[string]func(string) // 用户名 → 接收消息的回调
}

func (c *ChatRoom) Register(name string, handler func(string)) {
	c.users[name] = handler
}

func (c *ChatRoom) Broadcast(sender, msg string) {
	for name, h := range c.users {
		if name != sender {
			h("[" + sender + "] " + msg)
		}
	}
}
  • users 是 map 而非 slice:便于按用户名快速查找/移除
  • handler 类型为 func(string):解耦具体用户实现,调用方自己决定怎么处理消息
  • 不暴露 users 字段:防止外部绕过 Broadcast 直接调用

同事对象如何安全持有中介者引用

同事对象(比如 User)不能直接依赖具体中介者类型,否则会提高耦合度。推荐两种轻量方式:

  • 用函数字段替代结构体字段:onMessage func(string),由中介者在注册时注入
  • 若需双向通信(如用户主动发言),则持有一个指向中介者的指针:mediator *ChatRoom,但仅调用其公开方法

错误写法示例:mediator MediatorInterface —— Go 中接口应由调用方定义,而非中介者强推。

type User struct {
	name       string
	onMessage  func(string) // 收到消息时执行
	mediator   *ChatRoom    // 需要发言时用
}

func (u *User) Send(msg string) {
	if u.mediator != nil {
		u.mediator.Broadcast(u.name, msg)
	}
}

func (u *User) Receive(msg string) {
	if u.onMessage != nil {
		u.onMessage(msg)
	}
}

注意:onMessagemediator 不同时必需;根据协作方向选择其一即可。

什么时候不该用中介者模式

中介者容易变成“上帝对象”,尤其当所有状态变更都塞进一个结构体里时。以下情况建议跳过中介者:

  • 只有两个对象交互,且逻辑简单 → 直接互相调用更清晰
  • 协作规则随业务频繁变化 → 把逻辑写死在 Mediator 里会导致频繁修改,不如用事件总线(如 github.com/ThreeDotsLabs/watermill
  • 需要跨进程/跨服务协调 → 单机内存结构无法满足,应交由消息队列或状态机服务

真正值得上中介者的,是那些「对象数量中等(3–8 个)、交互关系密集、且生命周期基本一致」的模块,比如表单校验组件组、游戏内 NPC 行为调度器。


# git  # go  # github  # golang  # 为什么  # String  # 结构体  # 循环  # 指针  # 继承  # 接口  # class  # Struct  # 切片  # map  # 对象  # 事件  # uml  # ui  # 而非  # 放在  # 抽象类  # 什么时候  # 两种  # 会让  # 表单  # 死在  # 则会  # 应以 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Laravel如何处理异常和错误?(Handler示例)  Laravel项目怎么部署到Linux_Laravel Nginx配置详解  如何制作一个表白网站视频,关于勇敢表白的小标题?  Laravel如何使用Service Provider服务提供者_Laravel依赖注入与容器绑定【深度】  Laravel怎么在Blade中安全地输出原始HTML内容  如何在宝塔面板中修改默认建站目录?  Laravel怎么配置.env环境变量_Laravel生产环境敏感数据保护与读取【方法】  黑客如何利用漏洞与弱口令入侵网站服务器?  Laravel怎么判断请求类型_Laravel Request isMethod用法  长沙做网站要多少钱,长沙国安网络怎么样?  Laravel Eloquent性能优化技巧_Laravel N+1查询问题解决  QQ浏览器网页版登录入口 个人中心在线进入  使用豆包 AI 辅助进行简单网页 HTML 结构设计  潮流网站制作头像软件下载,适合母子的网名有哪些?  使用C语言编写圣诞表白程序  JavaScript如何实现错误处理_try...catch如何捕获异常?  VIVO手机上del键无效OnKeyListener不响应的原因及解决方法  Laravel Fortify是什么,和Jetstream有什么关系  javascript基本数据类型及类型检测常用方法小结  无锡营销型网站制作公司,无锡网选车牌流程?  青岛网站建设如何选择本地服务器?  Laravel如何正确地在控制器和模型之间分配逻辑_Laravel代码职责分离与架构建议  网站制作公司哪里好做,成都网站制作公司哪家做得比较好,更正规?  如何在IIS服务器上快速部署高效网站?  零服务器AI建站解决方案:快速部署与云端平台低成本实践  java获取注册ip实例  HTML5空格和margin有啥区别_空格与外边距的使用场景【说明】  Laravel如何创建自定义中间件?(Middleware代码示例)  实例解析angularjs的filter过滤器  香港服务器租用每月最低只需15元?  独立制作一个网站多少钱,建立网站需要花多少钱?  在线制作视频网站免费,都有哪些好的动漫网站?  Laravel如何实现API版本控制_Laravel API版本化路由设计策略  JavaScript数据类型有哪些_如何准确判断一个变量的类型  详解Android图表 MPAndroidChart折线图  Gemini怎么用新功能实时问答_Gemini实时问答使用【步骤】  Laravel模型事件有哪些_Laravel Model Event生命周期详解  Laravel辅助函数有哪些_Laravel Helpers常用助手函数大全  PHP 500报错的快速解决方法  Laravel怎么在Controller之外的地方验证数据  如何确保西部建站助手FTP传输的安全性?  Laravel怎么配置不同环境的数据库_Laravel本地测试与生产环境动态切换【方法】  如何实现javascript表单验证_正则表达式有哪些实用技巧  成都网站制作公司哪家好,四川省职工服务网是做什么用?  如何将凡科建站内容保存为本地文件?  为什么php本地部署后css不生效_静态资源加载失败修复技巧【技巧】  Laravel表单请求验证类怎么用_Laravel Form Request分离验证逻辑教程  网站制作软件有哪些,制图软件有哪些?  如何为不同团队 ID 动态生成多个独立按钮  Laravel如何清理系统缓存命令_Laravel清除路由配置及视图缓存的方法【总结】