当前位置: 首页 > news >正文

化繁为简:中介者模式如何管理复杂对象交互

化繁为简:中介者模式如何管理复杂对象交互

中介者模式 是一种行为型设计模式,定义了一个中介者对象,来封装一组对象之间的交互。中介者模式通过将对象之间的交互行为从多个对象中抽离出来,集中封装在一个中介者对象中,从而使各个对象之间的耦合松散,且可以独立地改变它们之间的交互。

核心思想

中介者模式通过引入一个中介者,协调多个对象之间的交互,避免对象之间的直接通信,从而减少对象之间的耦合

中介者模式的组成角色

Mediator(中介者接口)

  • 中介者定义了对象之间交互的接口,负责协调各个对象之间的通信行为。

ConcreteMediator(具体中介者)

  • 实现了 Mediator 接口,协调各具体同事对象之间的交互。它了解所有的同事对象,并通过其实现交互逻辑。

Colleague(同事类)

  • 每个同事对象只知道中介者,不能直接与其他同事对象通信。所有的交互都通过中介者来完成。

ConcreteColleague(具体同事类)

  • 实现了 Colleague 接口,依赖中介者与其他同事对象通信。每个具体同事对象的行为受到中介者的约束。

中介者模式的 UML 类图

在这里插入图片描述

类图解释

  1. Mediator(中介者接口):定义了同事对象之间的通信接口,通常定义 send() 方法来协调通信。
  2. ConcreteMediator(具体中介者)
    • 实现了 Mediator 接口,具体负责管理和协调同事对象之间的通信。它知道所有的具体同事对象,并通过 send() 方法传递消息或协调操作。
    • 它也可以持有具体的同事对象的引用,负责管理这些对象。
  3. Colleague(同事类):抽象类或接口,每个同事对象依赖中介者与其他同事进行交互。它不直接与其他同事对象通信,而是通过中介者通信。
  4. ConcreteColleague1、ConcreteColleague2(具体同事类):实现 Colleague,定义各自的具体操作。同事对象通过中介者协调相互间的通信

工作原理

  1. 集中通信:中介者模式通过引入 Mediator,将多个对象之间的交互逻辑集中到中介者中,使得各个同事类之间的关系变得松散耦合。
  2. 消息传递:同事对象通过调用中介者的 send() 方法进行通信。中介者负责接收和转发消息,并协调其他同事对象的行为。
  3. 解耦同事对象:同事对象之间不直接交互,而是通过中介者来处理。这样一来,各个对象之间的依赖关系被削弱了。

案例:机场塔台与飞机的调度

场景说明

在机场,飞机起飞和降落的调度是一个非常复杂的过程。如果每架飞机都直接与其他飞机进行通信,会导致混乱并增加空中事故的风险。因此,机场有一个控制塔(塔台),它负责协调所有飞机的起飞和降落

控制塔是中介者,飞机之间的所有通信都必须通过控制塔进行。飞机通过塔台来获取是否可以起飞或降落,从而避免与其他飞机发生冲突。

中介者模式的角色在机场塔台场景中的映射:

  1. Mediator(中介者接口):机场控制塔,负责协调所有飞机的起飞和降落。
  2. ConcreteMediator(具体中介者):具体实现的控制塔,协调飞机的飞行状态。
  3. Colleague(同事类):抽象飞机类,代表参与通信的对象(飞机)。
  4. ConcreteColleague(具体同事类):具体的飞机类,通过控制塔来决定起飞或降落。

在这里插入图片描述

代码实现:机场塔台调度系统

Step 1: 定义中介者接口

中介者接口定义了调度方法,用于协调飞机之间的通信。

// 中介者接口:塔台
public interface Mediator {void notify(String message, Airplane airplane);
}

Step 2: 实现具体的中介者

ControlTower 类是具体的中介者,它负责接收飞机的状态并通知其他飞机,确保起飞和降落顺序的安全。

import java.util.ArrayList;
import java.util.List;// 具体中介者:控制塔
public class ControlTower implements Mediator {private List<Airplane> airplanes;public ControlTower() {this.airplanes = new ArrayList<>();}public void registerAirplane(Airplane airplane) {airplanes.add(airplane);}@Overridepublic void notify(String message, Airplane airplane) {for (Airplane a : airplanes) {if (a != airplane) {a.receive(message);  // 通知其他飞机}}}
}

Step 3: 定义飞机抽象类

飞机类通过中介者来发送和接收信息,而不直接与其他飞机通信。

// 飞机抽象类
public abstract class Airplane {protected Mediator mediator;protected String name;public Airplane(Mediator mediator, String name) {this.mediator = mediator;this.name = name;}public abstract void send(String message);public abstract void receive(String message);
}

Step 4: 实现具体飞机类

具体飞机类实现了 send()receive() 方法,通过塔台来发送和接收消息。

Boeing737 类

// 具体飞机:Boeing 737
public class Boeing737 extends Airplane {public Boeing737(Mediator mediator, String name) {super(mediator, name);}@Overridepublic void send(String message) {System.out.println(this.name + " sends message: " + message);mediator.notify(message, this);  // 通过塔台发送消息}@Overridepublic void receive(String message) {System.out.println(this.name + " receives message: " + message);}
}

AirbusA320 类

// 具体飞机:Airbus A320
public class AirbusA320 extends Airplane {public AirbusA320(Mediator mediator, String name) {super(mediator, name);}@Overridepublic void send(String message) {System.out.println(this.name + " sends message: " + message);mediator.notify(message, this);  // 通过塔台发送消息}@Overridepublic void receive(String message) {System.out.println(this.name + " receives message: " + message);}
}

Step 5: 测试中介者模式

通过 ControlTower 来协调 Boeing737AirbusA320 之间的通信,确保飞机的起飞和降落顺序。

public class MediatorPatternDemo {public static void main(String[] args) {// 创建控制塔ControlTower tower = new ControlTower();// 创建飞机并注册到塔台Airplane boeing737 = new Boeing737(tower, "Boeing 737");Airplane airbusA320 = new AirbusA320(tower, "Airbus A320");tower.registerAirplane(boeing737);tower.registerAirplane(airbusA320);// 飞机发送消息boeing737.send("Requesting permission to land.");airbusA320.send("Requesting permission to take off.");}
}

输出结果

Boeing 737 sends message: Requesting permission to land.
Airbus A320 receives message: Requesting permission to land.
Airbus A320 sends message: Requesting permission to take off.
Boeing 737 receives message: Requesting permission to take off.

中介者模式在机场塔台场景中的工作原理

  1. 集中通信:所有的飞机(同事类)不直接通信,而是通过控制塔(中介者)进行协调。每架飞机都通过塔台发送和接收消息。
  2. 消息传递:当一架飞机发送起飞或降落请求时,塔台负责通知其他飞机,并保证飞行安全和秩序。
  3. 降低耦合性:飞机之间不直接通信,所有交互通过塔台完成,从而降低了飞机之间的耦合,增强了系统的灵活性和扩展性。

SpringMVC 中的 应用

Spring Framework 中,DispatcherServlet 作为前端控制器,实际上也承担了类似于中介者的作用。它在 Spring MVC 架构中负责协调不同组件(如控制器、视图解析器和处理器)的交互。

原理
  • DispatcherServlet 作为一个统一的请求分发器,负责接收 HTTP 请求,并根据请求的 URL 或其他条件,将请求分发给适当的处理器(如 @Controller 的方法)。它并不直接参与具体的请求处理,而是协调多个组件来完成请求-响应的流程。
  • 控制器、视图解析器、处理器等组件之间不直接通信,而是通过 DispatcherServlet 进行交互。
// DispatcherServlet 伪代码,简化示例
public class DispatcherServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {// 根据请求查找对应的控制器Handler handler = getHandler(request);// 调用控制器处理请求ModelAndView mv = handler.handleRequest(request, response);// 解析视图并返回响应render(mv, response);}
}

在这里,DispatcherServlet 类似于中介者,协调多个组件之间的交互,不同的控制器和视图解析器不直接互相通信,而是通过 DispatcherServlet 进行处理。

中介者模式的优缺点与应用场景总结

优点:

  1. 降低对象之间的耦合性
    • 中介者模式通过引入中介者,使多个对象之间的依赖关系转变为与中介者的依赖,减少了对象之间的直接交互,降低了系统的耦合度。
  2. 简化对象之间的通信
    • 所有通信都通过中介者集中管理,使得系统结构更加清晰,逻辑集中,尤其在多方通信的复杂系统中,简化了交互逻辑。
  3. 符合开闭原则
    • 中介者模式可以方便地扩展或修改交互逻辑,而不影响各个同事对象。可以通过修改中介者来增加或调整各对象之间的交互。
  4. 提高系统的灵活性
    • 当需要增加新的对象或修改对象之间的交互行为时,可以通过中介者进行控制,增强了系统的扩展性和灵活性。

缺点:

  1. 中介者可能变得复杂
    • 随着系统中同事对象的增多,中介者需要处理的交互逻辑也会变得越来越复杂,可能导致中介者变得难以维护,形成一个“上帝对象”。
  2. 不适合简单场景
    • 如果对象之间的交互非常简单,使用中介者模式会引入不必要的复杂性,增加维护成本。

应用场景:

  1. 复杂对象交互的系统
    • 在多个对象之间存在复杂交互的系统中,中介者模式非常适用。例如,GUI 控件交互(如按钮、文本框、下拉菜单等)可以通过中介者来管理。
  2. 消息队列系统
    • 在消息队列(如 JMS)中,中介者管理生产者和消费者之间的消息传递,解耦了消息的发送和接收,避免直接通信。
  3. 事件驱动系统
    • 在事件驱动架构中,事件发布者和监听者之间的通信可以通过中介者来协调,避免对象之间的紧密耦合。例如,Spring 的事件驱动机制利用 ApplicationEventPublisherApplicationListener 解耦了事件发布和处理。
  4. 航空管制系统
    • 像机场塔台这种需要协调多架飞机起降的系统,中介者可以集中管理对象的行为,保证复杂交互的有序进行。

相关文章:

化繁为简:中介者模式如何管理复杂对象交互

化繁为简&#xff1a;中介者模式如何管理复杂对象交互 中介者模式 是一种行为型设计模式&#xff0c;定义了一个中介者对象&#xff0c;来封装一组对象之间的交互。中介者模式通过将对象之间的交互行为从多个对象中抽离出来&#xff0c;集中封装在一个中介者对象中&#xff0c;…...

控制STM32蜂鸣器示例代码(江科大)

以下代码来源于本人学习江科大的课程&#xff0c;这是一个简单的STM32微控制器程序&#xff0c;用于控制连接到GPIOB第12号引脚的蜂鸣器。程序通过GPIOB的第12号引脚输出PWM波形来控制蜂鸣器的频率&#xff0c;从而产生声音。 #include "stm32f10x.h" …...

Java基础知识扫盲

目录 Arrays.sort的底层实现 BigDecimal(double)和BigDecimal(String)有什么区别 Char可以存储一个汉字吗 Java中的Timer定时调度任务是咋实现的 Java中的序列化机制是咋实现的 Java中的注解是干嘛的 Arrays.sort的底层实现 Arrays.sort是Java中提供的对数组进行排序的…...

ZLMediaKit Windows编译以及使用

1.运行ZLMediaKit 2.通过ffmpeg把视频源推流给ZLMediaKit 执行以下命令&#xff0c;将本地视频通过RTSP协议推流给ZLMediaKit。 ffmpeg -re -stream_loop -1 -i "D:\workplace\armgb\public\1.fileh264" -vcodec h264 -f rtsp rtsp://127.0.0.1/live/test 若想将本…...

基于YOLOv5s的无人机航拍输电线瓷瓶检测(附数据集与操作步骤)

本文主要内容:详细介绍了无人机航拍输电线瓷瓶检测的整个过程&#xff0c;从创建数据集到训练模型再到预测结果全部可视化操作与分析。 文末有数据集获取方式&#xff0c;请先看检测效果 现状 输电线路绝缘瓷瓶的检测主要依赖人工巡检。巡检人员需携带专业设备&#xff0c;攀…...

【Python百日进阶-Web开发-FastAPI】Day805 - FastAPI的请求体

文章目录 一、导入 Pydantic 的 BaseModel二、创建数据模型三、声明为参数四、结果五、自动化文档六、编辑器支持七、使用模型八、请求体 + 路径参数九、请求体 + 路径参数 + 查询参数十、不使用 Pydantichttps://fastapi.tiangolo.com/zh/tutorial/body/ 当你需要将数据从客户…...

【Kubernetes】常见面试题汇总(二十八)

目录 79.您如何看待公司从单一服务转向微服务并部署其服务容器&#xff1f; 80.什么是 Headless Service&#xff1f; 特别说明&#xff1a; 题目 1-68 属于【Kubernetes】的常规概念题。 题目 69-113 属于【Kubernetes】的生产应用题。 79.您如何看待公司从单一服务转…...

单ISP与双ISP的区别是什么

单ISP&#xff08;单一互联网服务提供商&#xff09;与双ISP&#xff08;双重互联网服务提供商&#xff09;主要在以下几个方面有区别&#xff1a; 服务冗余&#xff1a; 单ISP&#xff1a;只有一个互联网服务提供商提供的网络连接。如果该ISP发生故障&#xff0c;整个网络连接…...

【linux】gcc makefile

&#x1f525;个人主页&#xff1a;Quitecoder &#x1f525;专栏&#xff1a;linux笔记仓 目录 01.gcc如何完成02.gcc选项03.函数库与动静态链接静态链接动态链接库文件特点和用途动态链接版本和兼容性 04.makefile自动推导 01.gcc如何完成 预处理(进行宏替换) 预处理功能主要…...

12.Java基础概念-面向对象-static

欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 Facts speak louder than words&#xff01; 一、static关键字的含义…...

移动开发(三):使用.NET MAUI打包第一个安卓APK完整过程

目录 一、修改AndroidManifest.xml 配置APP基本信息权限 二、修改项目属性调整输出Android包格式为APK 三、项目发布 四、APP分发 五、总结 之前给大家介绍过使用使用.NET MAUI开发第一个安卓APP,今天给大家介绍如何打包成APK,然后安装到安卓手机正常运行。这里还是沿用…...

数据库中的DUAL表

在日常的数据库操作中&#xff0c;DUAL表是一个特殊的存在。它是一个伪表&#xff0c;用于在不需要实际数据表的情况下进行简单的查询。特别是在执行一些无关联的数据计算时&#xff0c;DUAL表经常派上用场。 什么是DUAL表&#xff1f; DUAL表最早出现在Oracle数据库中&#…...

寄宿制学校自闭症教育:为每个孩子创造奇迹

寄宿制学校自闭症教育&#xff1a;星贝育园——为每个孩子创造奇迹 在自闭症儿童教育的广阔领域中&#xff0c;寄宿制学校以其独特的教育模式和全方位的关怀体系&#xff0c;正逐步成为推动这些特殊孩子成长与发展的重要力量。广州的星贝育园自闭症儿童寄宿制学校&#xff0c;…...

Vue前端无法接收到后端返回的数据以及全局CSS样式影响(已解决)

Vue前端无法接收到后端返回的数据 前提&#xff1a;把很久以前的项目&#xff0c;翻出来重新优化一下&#xff0c;做一下前端的美化&#xff08;以前都是用的element的UI&#xff0c;现在想自己写&#xff09;。 由于是自己利用简单的html语句&#xff0c;主要面向各个按钮控…...

力扣234 回文链表 Java版本

文章目录 题目描述代码 题目描述 给你一个单链表的头节点 head &#xff0c;请你判断该链表是否为 回文链表 。如果是&#xff0c;返回 true &#xff1b;否则&#xff0c;返回 false 。 示例 1&#xff1a; 输入&#xff1a;head [1,2,2,1] 输出&#xff1a;true 示例 2&…...

银行性能测试怎么做?来认识下这4个性能测试工具!

一、银行项目性能测试是什么&#xff1f; 银行项目性能测试是一种软件测试形式&#xff0c;用于评估银行应用程序的性能和负载能力。 它的主要目标是确保银行系统能够在不同负载水平下提供稳定、高效的服务。 性能测试涵盖了多个方面&#xff0c;包括响应时间、吞吐量、并发用…...

FME学习笔记

读取数据 方法一&#xff1a;add reader 通过读模块来进行数据的读取 方法二&#xff1a;FeatureReader Parameters 通过转换器来进行数据的读取 可以通过空间范围进行筛选 在FME中&#xff0c;所有数据处理都要用到的&#xff0c;绝对的重点&#xff1a;转换器&#xff…...

机器翻译之创建Seq2Seq的编码器、解码器

1.创建编码器、解码器的基类 1.1创建编码器的基类 from torch import nn#构建编码器的基类 class Encoder(nn.Module): #继承父类nn.Moduledef __init__(self, **kwargs): #**kwargs&#xff1a;不定常的关键字参数super().__init__(**kwargs)def forward(self, X, *args…...

锤炼核心技能以应对编程革命

一、引言  随着人工智能的快速发展&#xff0c;尤其是AIGC等大语言模型的涌现&#xff0c;AI辅助编程工具逐渐成为程序员的新伙伴。这一变革不仅引发了关于AI是否能取代部分编程工作的讨论&#xff0c;也促使程序员重新思考自己的职业发展和技能提升路径。在AI时代&#xff0…...

2024 go-zero社交项目实战

背景 一位商业大亨&#xff0c;他非常看好国内的社交产品赛道&#xff0c;想要造一款属于的社交产品&#xff0c;于是他找到了负责软件研发的小明。 小明跟张三一拍即合&#xff0c;小明决定跟张三大干一番。 社交产品MVP版本需求 MVP指&#xff1a;Minimum Viable Product&…...

【大模型RAG】拍照搜题技术架构速览:三层管道、两级检索、兜底大模型

摘要 拍照搜题系统采用“三层管道&#xff08;多模态 OCR → 语义检索 → 答案渲染&#xff09;、两级检索&#xff08;倒排 BM25 向量 HNSW&#xff09;并以大语言模型兜底”的整体框架&#xff1a; 多模态 OCR 层 将题目图片经过超分、去噪、倾斜校正后&#xff0c;分别用…...

Cilium动手实验室: 精通之旅---20.Isovalent Enterprise for Cilium: Zero Trust Visibility

Cilium动手实验室: 精通之旅---20.Isovalent Enterprise for Cilium: Zero Trust Visibility 1. 实验室环境1.1 实验室环境1.2 小测试 2. The Endor System2.1 部署应用2.2 检查现有策略 3. Cilium 策略实体3.1 创建 allow-all 网络策略3.2 在 Hubble CLI 中验证网络策略源3.3 …...

c++ 面试题(1)-----深度优先搜索(DFS)实现

操作系统&#xff1a;ubuntu22.04 IDE:Visual Studio Code 编程语言&#xff1a;C11 题目描述 地上有一个 m 行 n 列的方格&#xff0c;从坐标 [0,0] 起始。一个机器人可以从某一格移动到上下左右四个格子&#xff0c;但不能进入行坐标和列坐标的数位之和大于 k 的格子。 例…...

css的定位(position)详解:相对定位 绝对定位 固定定位

在 CSS 中&#xff0c;元素的定位通过 position 属性控制&#xff0c;共有 5 种定位模式&#xff1a;static&#xff08;静态定位&#xff09;、relative&#xff08;相对定位&#xff09;、absolute&#xff08;绝对定位&#xff09;、fixed&#xff08;固定定位&#xff09;和…...

关于 WASM:1. WASM 基础原理

一、WASM 简介 1.1 WebAssembly 是什么&#xff1f; WebAssembly&#xff08;WASM&#xff09; 是一种能在现代浏览器中高效运行的二进制指令格式&#xff0c;它不是传统的编程语言&#xff0c;而是一种 低级字节码格式&#xff0c;可由高级语言&#xff08;如 C、C、Rust&am…...

RNN避坑指南:从数学推导到LSTM/GRU工业级部署实战流程

本文较长&#xff0c;建议点赞收藏&#xff0c;以免遗失。更多AI大模型应用开发学习视频及资料&#xff0c;尽在聚客AI学院。 本文全面剖析RNN核心原理&#xff0c;深入讲解梯度消失/爆炸问题&#xff0c;并通过LSTM/GRU结构实现解决方案&#xff0c;提供时间序列预测和文本生成…...

ip子接口配置及删除

配置永久生效的子接口&#xff0c;2个IP 都可以登录你这一台服务器。重启不失效。 永久的 [应用] vi /etc/sysconfig/network-scripts/ifcfg-eth0修改文件内内容 TYPE"Ethernet" BOOTPROTO"none" NAME"eth0" DEVICE"eth0" ONBOOT&q…...

API网关Kong的鉴权与限流:高并发场景下的核心实践

&#x1f525;「炎码工坊」技术弹药已装填&#xff01; 点击关注 → 解锁工业级干货【工具实测|项目避坑|源码燃烧指南】 引言 在微服务架构中&#xff0c;API网关承担着流量调度、安全防护和协议转换的核心职责。作为云原生时代的代表性网关&#xff0c;Kong凭借其插件化架构…...

《Docker》架构

文章目录 架构模式单机架构应用数据分离架构应用服务器集群架构读写分离/主从分离架构冷热分离架构垂直分库架构微服务架构容器编排架构什么是容器&#xff0c;docker&#xff0c;镜像&#xff0c;k8s 架构模式 单机架构 单机架构其实就是应用服务器和单机服务器都部署在同一…...

java高级——高阶函数、如何定义一个函数式接口类似stream流的filter

java高级——高阶函数、stream流 前情提要文章介绍一、函数伊始1.1 合格的函数1.2 有形的函数2. 函数对象2.1 函数对象——行为参数化2.2 函数对象——延迟执行 二、 函数编程语法1. 函数对象表现形式1.1 Lambda表达式1.2 方法引用&#xff08;Math::max&#xff09; 2 函数接口…...