设计模式-中介者(调停者)模式(行为型)
中介者模式
中介者模式是一种行为型模式,又叫调停者模式,它是为了解决多个对象之间,多个类之间通信的复杂性,定义一个中介者对象来封装一些列对象之间的交互,使各个对象之间不同持有对方的引用就可以实现交互,降低耦合度;实际开发中,消息队列、服务注册中心、MVC框架中的controller都是中介者;
图解

角色
- 同事对象:定义抽象接口,用于与中介者进行通信,一般是一个发送消息的接口,一个接收消息的接口
- 具体同事对象:实现同事对象抽象接口,具体发送、接收消息的逻辑
- 中介者:定义要给抽象接口用于与同事对象进行通信
- 具体中介者:实现同事之间通信的逻辑
案例1-中介者持有同事对象的引用
中介者
public interface Mediator {void execute();
}
/** 实现1*/
public class AllPersonMediator implements Mediator {private List<Person> personList;public AllPersonMediator(List<Person> personList) {this.personList = personList;}@Overridepublic void execute() {System.out.println("开家长会啦:");personList.forEach(item -> System.out.println(item.getName()));}
}
/** 实现2*/
public class TeacherByTeacherMediator implements Mediator {private Person p1;private Person p2;public TeacherByTeacherMediator(Teacher p1, Teacher p2) {this.p1 = p1;this.p2 = p2;}@Overridepublic void execute() {System.out.println(p1.getName()+"和" + p2.getName() + "两个老师之间的协同的任务");}
}
/** 实现3*/
public class TeacherAndStudentAndParentsMediator implements Mediator {private Person p1;private Person p2;private Person p3;public TeacherAndStudentAndParentsMediator(Teacher p1, Student p2, Parents p3) {this.p1 = p1;this.p2 = p2;this.p3 = p3;}@Overridepublic void execute() {System.out.println("老师"+p1.getName()+"通知家长"+p3.getName() +"学生" +p2.getName()+ "成绩不及格!");}
}
同事类:
public abstract class Person {private String name;public Person(String name) {this.name = name;}public String getName() {return name;}
}public class Student extends Person {public Student(String name) {super(name);}
}public class Student extends Person {public Student(String name) {super(name);}
}public class Parents extends Person {public Parents(String name) {super(name);}
}
测试类:
public class Test01 {public static void main(String[] args) {Teacher t1 = new Teacher("王老师");Teacher t2 = new Teacher("刘老师");Student s1 = new Student("小明");Parents p1 = new Parents("小明爸爸");TeacherByStudentMediator teacherByTeacherMediator = new TeacherByStudentMediator(t1, s1);teacherByTeacherMediator.execute();TeacherAndStudentAndParentsMediator teacherAndStudentAndParentsMediator = new TeacherAndStudentAndParentsMediator(t1, s1, p1);teacherAndStudentAndParentsMediator.execute();List<Person> list = new ArrayList<>();list.add(t1);list.add(t2);list.add(s1);list.add(p1);AllPersonMediator allPersonMediator = new AllPersonMediator(list);allPersonMediator.execute();}
}
案例2-同事类持有中介者的引用
中介者
public interface SimpleMediator {Queue<Client> queue = new ConcurrentLinkedQueue<>();void add(Client c);Client get();
}
/** 实现*/
public class MessageSimpleMediator implements SimpleMediator{@Overridepublic void add(Client c) {queue.add(c);}@Overridepublic Client get() {return queue.poll();}
}
同事类
public abstract class Client<T> {protected String name;protected T message;protected SimpleMediator simpleMediator;public Client(String name, T message, SimpleMediator simpleMediator) {this.name = name;this.message = message;this.simpleMediator = simpleMediator;}public Client(String name, SimpleMediator simpleMediator) {this.name = name;this.simpleMediator = simpleMediator;}/** 发送消息*/abstract void sent();/** 获取消息*/abstract void get();public String getName() {return name;}public void setName(String name) {this.name = name;}public T getMessage() {return message;}public void setMessage(T message) {this.message = message;}
}public class ReceiverClient extends Client<String>{public ReceiverClient(String name, SimpleMediator simpleMediator) {super(name, simpleMediator);}@Overridevoid sent() {new RuntimeException("这是接收消息的客户端");}@Overridevoid get() {Client client = simpleMediator.get();System.out.println("发送人:" + client.getName());System.out.println("发送内容:" + client.getMessage());}
}public class SenderClient extends Client<String>{public SenderClient(String name, String message, SimpleMediator simpleMediator) {super(name, message, simpleMediator);}@Overridevoid sent() {simpleMediator.add(this);System.out.println("消息发送成功!"); }@Overridevoid get() {new RuntimeException("这是接收消息的客户端");}
}
测试类:
public class Test02 {public static void main(String[] args) {MessageSimpleMediator messageSimpleMediator = new MessageSimpleMediator();SenderClient C1 = new SenderClient("发送消息的客户端1", "发送的内容", messageSimpleMediator);SenderClient C2 = new SenderClient("发送消息的客户端2", "发送的内容", messageSimpleMediator);SenderClient C3 = new SenderClient("发送消息的客户端3", "发送的内容", messageSimpleMediator);ReceiverClient R3 = new ReceiverClient("接收消息的客户端", messageSimpleMediator);C1.sent();C2.sent();C3.sent();while (messageSimpleMediator.queue.size() > 0){R3.get();}}
}
总结
在学习中介者模式的时候一直疑惑,为什么要有中介者接口?
学习完后感悟:设计模式是一种思想,不应该去套公式。中介者模式的核心思想就是通过对象引用的方式实现多个同事类通过一个中介者建立联系,可以中介者中引用同事类,也可以同时类引用中介者,甚至可以用一个不用中介者接口,直接定义具体中介者实现。具体怎么实现要根据具体情况而论。
设计模式是一种思想,而不是一种公式模板。
相关文章:
设计模式-中介者(调停者)模式(行为型)
中介者模式 中介者模式是一种行为型模式,又叫调停者模式,它是为了解决多个对象之间,多个类之间通信的复杂性,定义一个中介者对象来封装一些列对象之间的交互,使各个对象之间不同持有对方的引用就可以实现交互…...
HC-05蓝牙模块配置连接和使用
文章目录 1. 前期准备 2. 进入AT模式 3. 电脑串口配置 4. 配置过程 5. 主从机蓝牙连接 6. 蓝牙模块HC-05和电脑连接 1. 前期准备 首先需要准备一个USB转TTL连接器,电脑安装一个串口助手,然后按照下面的连接方式将其相连。 VCCVCCGNDGNDRXDTXDTXD…...
云上小知识:企业选择云服务的小Tips
企业在选择云服务模式时,应综合考虑以下几个关键因素: 1. 业务需求与场景 企业需要根据自身的业务特点和需求来选择合适的云服务模式。例如,如果企业的用户分布广泛,需要跨地域提供服务,那么公有云可能是更好的选择。…...
生成式人工智能 - Stable Diffusion 都使用了哪些技术?
一、Stable Diffusion简述 1、简述 Stable Diffusion在2022年8月开源,是由慕尼黑大学的CompVis研究团队开发的生成式人工神经网络。该项目由初创公司StabilityAI、CompVis和Runway合作开发,并得到了EleutherAI和LAION的支持。截至2022年10月,StabilityAI已筹集了1.01亿美元…...
React的useState的基础使用
import {useState} from react // 1.调用useState添加状态变量 // count 是新增的状态变量 // setCount 修改状态变量的方法 // 2.添加点击事件回调 // userState实现计数实例import {useState} from react// 使用组件 function App() {// 1.调用useState添加状态变量// coun…...
接口自动化Requests+Pytest基础实现
目录 1. 数据库以及数据库操作1.1 概念1.2 分类1.3 作用 2 python操作数据库的相关实现2.1 背景2.2 相关实现 3. pymysql基础3.1 整个流程3.2 案例3.3 Pymysql工具类封装 4 事务4.1 案例4.2 事务概念4.3 事务特征 5. requests库5.1 概念5.2 角色定位5.3 安装5.4 校验5.5 reques…...
深入解析Kafka消息传递的可靠性保证机制
深入解析Kafka消息传递的可靠性保证机制 Kafka在设计上提供了不同层次的消息传递保证,包括at most once(至多一次)、at least once(至少一次)和exactly once(精确一次)。每种保证通过不同的机制…...
jEasyUI 设置排序
jEasyUI 设置排序 jEasyUI 是一个基于 jQuery 的框架,用于轻松构建交互式的 Web 应用程序。它提供了一系列的 UI 组件,如表格(datagrid)、树(tree)、下拉列表(combobox)等,这些组件可以帮助开发者快速实现复杂的界面功能。在本文中,我们将重点讨论如何在 jEasyUI 中…...
MySQL之查询性能优化(十二)
查询性能优化 优化COUNT()查询 4.使用近似值 有时候某些业务场景并不要求完全精确的COUNT值,此时可以用近似值来代替。EXPLAIN出来的优化器估算的行数就是一个不错的近似值,执行EXPLAIN并不需要真正地去执行查询,所以成本很低。很多时候&am…...
7-16 二分查找
7-16 二分查找 分数 25 全屏浏览 切换布局 作者 李廷元 单位 中国民用航空飞行学院 请实现有重复数字的有序数组的二分查找。 输出在数组中第一个大于等于查找值的位置,如果数组中不存在这样的数,则输出数组长度加一。 输入格式: 输入第一行有两个…...
对Java中二维数组的深层认识
首先,在JAVA中,二维数组是一种数组的数组。它可以看作是一个矩阵,通常是由于表示二维数据节后,如表格和网格。 1.声明和初始化二维数组 声明 int[][] arr;初始化 int[][] arrnew int[3][4];或者用花括号嵌套 int[][] arr{{1,…...
C++的STL 中 set.map multiset.multimap 学习使用详细讲解(含配套OJ题练习使用详细解答)
目录 一、set 1.set的介绍 2.set的使用 2.1 set的模板参数列表 2.2 set的构造 2.3 set的迭代器 2.4 set的容量 2.5 set的修改操作 2.6 set的使用举例 二、map 1.map的介绍 2.map的使用 2.1 map的模板参数说明 2.2 map的构造 2.3 map的迭代器 2.4 map的容量与元…...
【Java笔记】第10章:接口
前言1. 接口的概念与定义2. 接口的声明与语法3. 接口的实现4. 接口的继承5. 接口的默认方法6. 接口的静态方法7. 接口的私有方法8. 接口的作用9. 接口与抽象类的区别10. 接口在Java集合中的应用结语 上期回顾:【Java笔记】第9章:三个修饰符 个人主页:C_G…...
Angular知识概览
Angular 是一个由 Google 维护的开源前端框架,用于构建动态网页应用。以下是对 Angular 主要概念和特性的概览: 1. Angular 的核心概念 - 组件 (Component):Angular 应用的基本构建块。每个组件包括一个 TypeScript 类,用于处理数…...
经典文献阅读之--Online Monocular Lane Mapping(使用Catmull-Rom样条曲线完成在线单目车道建图)
0. 简介 对于单目摄像头完成SLAM建图这类操作,对于自动驾驶行业非常重要,《Online Monocular Lane Mapping Using Catmull-Rom Spline》介绍了一种仅依靠单个摄像头和里程计生成基于样条的在线单目车道建图方法。我们提出的技术将车道关联过程建模为一个…...
frida timed out
从Android Q(10)开始,Google引入了一种新的机制,加快了app的启动时间 Android USAP 进程启动流程 adb shell su ps -A | grep usaproot 9917 1032 6577052 13676 __skb_wait_for_more_packets 0 S usap64 root 9928 1032 6577052…...
51单片机-独立按键控制灯灯灯
目录 简介: 一. 1个独立按钮控制一个灯例子 二. 在加一个独立按键,控制第二个灯 三. 第一个开关 开灯, 第二个开关关灯 四. 点一下开灯,在点一下关灯 五. 总结 简介: 51 单片机具有强大的控制能力,而独立按键则提供了一种简单的输入方式。 当把独立按键与 …...
【C++】用红黑树封装map、set
用红黑树封装map、set 1. 红黑树1.1 模板参数的控制1.1.1 Value1.1.2 KeyOfValue 1.2 正向迭代器1.2.1 构造函数1.2.2 begin()end()1.2.3 operator()1.2.4 operator--()1.2.5 operator*()1.2.6 operator->()1.2.7 operator()1.2.8 operator!()1.2.9 总代码 1.3 反向迭代器1.…...
【中颖】SH79F9202 串口通信
头文件 uart.h #ifndef UART_H #define UART_H#include "SH79F9202.h" #include "LCD.h" #include "timer2.h" #include "timer5.h" #include "cpu.h" #include "key.h" #include "io.h" #include &qu…...
IDEA创建Maven项目
IDEA创建Maven项目 第一步:创建新项目 或者 第二步:创建maven模块 前提条件: File>>Settings,检查自己的maven是否已经安装配置好 创建maven模块 其中Archetype一般选择如下 点击创建后生成如下 需要在main目录下创…...
Cadence IC617实战:VerilogA vs analogLib搭建全差分放大器,哪个更适合你?
Cadence IC617实战:VerilogA与analogLib全差分放大器设计深度对比 在模拟IC设计领域,全差分放大器作为基础构建模块,其实现方式直接影响设计效率和仿真精度。Cadence IC617作为行业标准工具,提供了VerilogA和analogLib两种截然不同…...
GluonCV版本升级指南:从0.8到0.11的10大新特性详解
GluonCV版本升级指南:从0.8到0.11的10大新特性详解 【免费下载链接】gluon-cv dmlc/gluon-cv: GluonCV 是由DMLC(Apache MXNet背后的社区)开发的一个计算机视觉库,为研究人员和工程师提供了大量预训练模型、基准测试和工具&#x…...
ZeroOmega代理规则引擎:构建智能化网络访问策略
ZeroOmega代理规则引擎:构建智能化网络访问策略 【免费下载链接】ZeroOmega Manage and switch between multiple proxies quickly & easily. 项目地址: https://gitcode.com/gh_mirrors/ze/ZeroOmega 在数字化生活中,我们每天都在与各种网络…...
Terraria 源代码架构解析:从核心功能到启动配置的全方位指南
Terraria 源代码架构解析:从核心功能到启动配置的全方位指南 【免费下载链接】Terraria-Source-Code 项目地址: https://gitcode.com/gh_mirrors/te/Terraria-Source-Code Terraria 源代码项目是一款经典沙盒游戏的开源实现,包含了世界生成、实体…...
实战对比:Vamana/HNSW/NSG三大图算法在百维向量搜索中的性能差异
百维向量搜索实战:Vamana/HNSW/NSG三大图算法性能横评 在当今数据爆炸的时代,高效处理高维向量搜索已成为推荐系统、图像识别和自然语言处理等领域的核心技术瓶颈。面对百维甚至更高维度的向量数据,传统暴力搜索方法早已力不从心,…...
Qwen3-0.6B-FP8企业级部署教程:基于Dify打造AI应用平台
Qwen3-0.6B-FP8企业级部署教程:基于Dify打造AI应用平台 想快速搭建一个属于自己或团队的AI应用,但又觉得从零开发太复杂?今天,我们就来聊聊如何用Qwen3-0.6B-FP8这个轻量高效的模型,结合Dify这个强大的AI应用开发平台…...
ChatTTS 量化模型实战:从模型压缩到推理效率提升
最近在部署 ChatTTS 模型时,遇到了一个很实际的问题:模型虽然效果不错,但体积大、推理慢,在资源受限的边缘设备上跑起来非常吃力。显存动不动就占好几个G,生成一段语音的等待时间也让人着急。为了解决这个问题…...
C/C++中备受争议却难以替代的goto语句:效率与可读性的博弈
1. goto语句的前世今生 在C/C的世界里,goto就像是个"老古董"——它从1950年代的Fortran语言一路走来,至今仍在某些角落发光发热。我第一次在Linux内核代码里看到密密麻麻的goto时,整个人都懵了:这玩意儿不是教科书上明令…...
OpenClaw对接ollama模型:GLM-4.7-Flash接口配置详解
OpenClaw对接ollama模型:GLM-4.7-Flash接口配置详解 1. 为什么选择本地ollama部署GLM-4.7-Flash 去年我在尝试构建个人自动化工作流时,发现公有云API调用不仅费用高昂,还存在隐私顾虑。直到发现ollama这个轻量级模型运行框架,配…...
5步搞定OpenClaw+Qwen3-32B:RTX4090D镜像一键接入实战
5步搞定OpenClawQwen3-32B:RTX4090D镜像一键接入实战 1. 为什么选择云端沙盒方案 当我第一次听说OpenClaw这个开源自动化框架时,内心既兴奋又忐忑。作为一个喜欢折腾新技术的开发者,我迫不及待想尝试这个能像人类一样操作电脑的AI助手。但看…...
