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

设计模式: 行为型之中介者模式(18)

中介者模式概述

  • 中介者模式(Mediator Pattern)是一种行为设计模式,它用于减少对象之间的直接交互,从而使其可以松散耦合
  • 中介者模式通过引入一个中介者对象来协调多个对象之间的交互,使得这些对象不需要知道彼此的具体实现,只需与中介者进行通信
  • 中介者模式结构
    • 中介者(Mediator)
      • 定义了一个接口,用于与各同事对象通信
    • 具体中介者(ConcreteMediator)
      • 实现中介者接口,协调各个同事对象之间的交互
      • 并可以维护同事对象之间的引用关系
    • 同事类(Colleague)
      • 每一个同事对象都知道中介者对象,并与它通信
      • 但不知道其他同事对象,同事对象之间的交互通过中介者对象来完成

中介者模式应用

// 1. 定义同事类(Colleague)接口
interface Colleague {mediator: Mediator;send(message: string): void;receive(message: string): void;
}// 2. 定义中介者(Mediator)接口
interface Mediator {register(colleague: Colleague): void;sendMessage(sender: string, message: string): void;
}// 3. 创建具体同事类
class ConcreteColleagueA implements Colleague {mediator: Mediator;name: string;constructor(name: string, mediator: Mediator) {this.mediator = mediator;this.name = name;}send(message: string): void {this.mediator.sendMessage(this.name, message);}receive(message: string): void {console.log(`${this.name} received the message: ${message}`);}
}class ConcreteColleagueB implements Colleague {mediator: Mediator;name: string;constructor(name: string, mediator: Mediator) {this.mediator = mediator;this.name = name;}send(message: string): void {this.mediator.sendMessage(this.name, message);}receive(message: string): void {console.log(`${this.name} received the message: ${message}`);}
}// 4. 创建具体中介者类
class ConcreteMediator implements Mediator {private colleagues: Colleague[] = [];register(colleague: Colleague): void {this.colleagues.push(colleague);}sendMessage(sender: string, message: string): void {this.colleagues.forEach(colleague => {if (colleague.name !== sender) {colleague.receive(message);}});}
}// 使用示例
function main() {const mediator = new ConcreteMediator();const colleagueA = new ConcreteColleagueA("Colleague A", mediator);const colleagueB = new ConcreteColleagueB("Colleague B", mediator);mediator.register(colleagueA);mediator.register(colleagueB);colleagueA.send("Hello from A");colleagueB.send("Hello from B");
}main();
  • 首先,我们定义了两个接口:ColleagueMediator
    • Colleague 接口表示参与交互的对象,它有两个方法:send 用于发送消息,receive 用于接收消息
    • 并且有一个属性 mediator,指向中介者对象。Mediator 接口则定义了中介者应当具备的能力,包括注册同事对象(register)和转发消息(sendMessage
  • 我们创建了两个实现了 Colleague 接口的具体同事类 ConcreteColleagueAConcreteColleagueB
    • 每个具体同事类都有自己的名字,并且持有中介者对象
    • 当它们想要发送消息时,通过调用中介者的 sendMessage 方法而非直接和其他同事类通信
  • 实现了 Mediator 接口的 ConcreteMediator 类是具体的中介者
    • 它维护了一个同事对象列表,并提供 register 方法用于同事对象的注册
    • 当一个同事对象通过 sendMessage 发送消息时,中介者会遍历同事列表,将消息转发给除了发送者之外的所有同事对象
  • main 函数中,我们创建了一个 ConcreteMediator 对象作为中介者,并创建了两个同事对象(ConcreteColleagueAConcreteColleagueB
    • 接着,我们把这两个同事对象注册到中介者,并让每个同事对象发送一条消息
    • 此时,消息通过中介者转发,确保每个同事都能收到除自己发送之外的消息
  • 通过这个示例,我们可以看到中介者模式是如何解耦同事对象之间直接的通信关系,转而通过中介者来进行消息传递,从而降低了耦合度,增强了系统的可维护性和扩展性

中介者模式应用场景

  • 对象间的交互复杂:当系统中有多个对象相互协作,且它们之间的交互关系错综复杂,表现为网状结构或多对多关系时,可以使用中介者模式来集中管理这些交互,将原来分散在各个对象之间的交互逻辑转移到中介者中,从而简化对象间的直接依赖关系

  • 需要解耦对象:当发现系统中对象彼此之间的依赖性过高,修改一个对象可能引起一系列连锁反应时,通过中介者模式可以使对象之间的耦合度降低,提高系统的可维护性和扩展性

  • 通信逻辑需要集中控制:在一些场景中,如网络聊天室、飞行控制系统、游戏引擎中的实体交互、UI组件间通信等,需要有一个中心对象来管理和协调不同对象间的通信,这些场景非常适合采用中介者模式

  • 易于扩展新功能:当系统中未来可能增加更多参与者(对象)或者改变交互规则时,通过中介者模式可以相对容易地调整和扩展通信逻辑,因为大部分交互都在中介者内完成,不涉及其他对象的改动

  • 需要支持模块化:在大型软件系统中,中介者模式可以帮助将系统划分为可独立开发和测试的模块,每个模块作为一个对象与中介者交互,而模块间的复杂交互则通过中介者来组织和协调

  • 总之,中介者模式尤其适用于那些对象间交互繁杂,且希望避免对象之间紧密耦合的场景,通过引入中介者,可以将复杂的交互关系转化为中介者与各个对象之间的简单交互

中介者模式优缺点


1 ) 优点

  • 降低对象之间的耦合度:同事对象之间不再直接相互调用,而是通过中介者来间接通信,减少了它们之间的耦合

  • 易于维护:由于将对象之间的交互逻辑集中到了中介者对象中,因此如果需要修改交互方式,只需修改中介者即可,而无需修改所有同事对象

  • 简化对象之间的通信:同事对象不再需要知道其他对象的存在,只需与中介者通信,降低了通信的复杂性

  • 支持灵活的通信方式:中介者可以根据需要实现不同的通信策略,比如广播、单播等

2 )缺点

  • 可能产生过多的中介者:如果系统中存在过多的中介者,可能会导致系统结构变得复杂

  • 中介者可能变得过于复杂:如果中介者需要处理大量的同事对象之间的交互,其内部逻辑可能会变得相当复杂,难以维护

  • 可能引入新的依赖:同事对象依赖于中介者,如果中介者出现问题,可能会影响到多个同事对象的正常工作

  • 在使用中介者模式时,需要根据具体场景和需求权衡其优缺点,确保在降低耦合度和简化通信的同时,避免引入过多的复杂性和依赖关系

中介者模式和代理模式区别和联系

  • 中介者模式(Mediator Pattern)和代理模式(Proxy Pattern)都是设计模式中的行为型模式,它们在解决软件设计问题上有不同的侧重点和应用场景

1 )中介者模式

  • 目的: 解决对象之间的多重关联和通信复杂性问题,通过引入中介者对象来封装对象间的交互逻辑,使得原本相互耦合的对象转变为与中介者对象的单向关联,从而降低对象之间的耦合度。

  • 角色: 同事类(Colleague)、中介者(Mediator)

  • 作用: 中介者对象集中处理同事类之间的交互,同事类只与中介者交互,不直接与其他同事类交互。

  • 适用场景: 多个对象之间存在复杂的网状交互,如聊天室的消息路由、组件间的通信协调等

2 )代理模式

  • 目的: 为其他对象提供一个代理,以控制对这个对象的访问,增强或扩展功能,如延迟加载、安全性控制、访问权限控制、远程代理、虚拟代理等。
  • 角色: 主题(Subject)、代理(Proxy)、真实主题(Real Subject)
  • 作用: 代理对象替代真实对象,对外提供相同的接口,但可以在调用前后附加额外的操作,比如缓存、预处理、后处理、权限检查等。
  • 适用场景: 需要控制访问、增强功能、保护目标对象或透明地添加额外操作等场合,如数据库连接池、图片懒加载、远程服务调用等。

3 )区别与联系

  • 区别
    • 目的不同:中介者模式主要用于解耦对象间的复杂交互,而代理模式主要用于控制对对象的访问
    • 结构不同:中介者模式中对象与中介者单向交互,代理模式中代理与真实对象的接口一致,代理对象代替或辅助真实对象工作
    • 作用不同:中介者模式改变了对象间的通信结构,代理模式通常不会改变对象间的结构,只是在访问过程中增加了额外的逻辑
  • 联系
    • 都属于行为型设计模式,都通过引入中间层对象来改善原有对象结构和行为
    • 都是对对象功能的一种包装和扩展,只不过中介者模式更关注于整体的交互协调,而代理模式更侧重于个体对象的访问控制和功能扩展
    • 在实际开发中,这两种模式经常结合使用,共同优化系统的架构和设计

相关文章:

设计模式: 行为型之中介者模式(18)

中介者模式概述 中介者模式(Mediator Pattern)是一种行为设计模式,它用于减少对象之间的直接交互,从而使其可以松散耦合中介者模式通过引入一个中介者对象来协调多个对象之间的交互,使得这些对象不需要知道彼此的具体…...

计算机网络的起源与发展历程

文章目录 前言时代背景ARPANET 的诞生TCP/IP 协议簇与 Internet 的诞生HTTP 协议与 Web 世界结语 前言 在当今数字化时代,计算机网络已经成为我们生活中不可或缺的一部分。无论是在家庭、学校、还是工作场所,我们都能感受到网络的巨大影响。随着互联网的…...

2024-4-12-实战:商城首页(下)

个人主页:学习前端的小z 个人专栏:HTML5和CSS3悦读 本专栏旨在分享记录每日学习的前端知识和学习笔记的归纳总结,欢迎大家在评论区交流讨论! 文章目录 作业小结 作业 .bg-backward {width: 60px; height: 60px;background: url(..…...

一、flask入门和视图

run启动参数 模板渲染 后端给前端页面传参 前端页面设置css from flask import Flask, render_template,jsonify# 创建flask对象 app Flask(__name__)# 视图函数 路由route app.route("/") def hello_world():# 响应,返回给前端的数据return "h…...

Selenium+Chrome Driver 爬取搜狐页面信息

进行selenium包和chromedriver驱动的安装 安装selenium包 在命令行或者anaconda prompt 中输入 pip install Selenium 安装 chromedriver 先查看chrome浏览器的版本 这里是 123.0.6312.106 版 然后在http://npm.taobao.org/mirrors/chromedriver/或者https://googlechrom…...

SpringBoot:一个注解就能帮你下载任意对象

介绍 下载功能应该是比较常见的功能了,虽然一个项目里面可能出现的不多,但是基本上每个项目都会有,而且有些下载功能其实还是比较繁杂的,倒不是难,而是麻烦。 所以结合之前的下载需求,我写了一个库来简化…...

oracle全量、增量备份

采用0221222增量备份策略,7天一个轮回 也就是周日0级备份,周1 2 4 5 6 采用2级增量备份,周3采用1级增量备份 打开控制文件自动备份 CONFIGURE CONTROLFILE AUTOBACKUP ON; 配置控制文件备份路径 CONFIGURE CONTROLFILE AUTOBACKUP FORMAT FOR DEVI…...

React Router 5 vs 6:使用上的主要差异与升级指南

React Router 5 的一些API 在 React Router 6 上有时可能找不到,可能会看到如下画面:export ‘useHistory’ was not found in ‘react-router-dom’ … React Router目前有两个大的版本,即React Router 5、6。React Router 6 在设计上更加简…...

基于LNMP部署wordpress

目录 一.环境准备 二.配置源并安装 三.配置Nginx 四.配置数据库 五.上传源码并替换 六.打开浏览器,输入虚拟机ip访问安装部署 七.扩展增加主题 一.环境准备 centos7虚拟机 关闭防火墙和seliunx stop firewalld #关闭防火墙 setenforce 0 …...

openGauss_5.1.0 企业版快速安装及数据库连接:单节点容器化安装

目录 📚第一章 官网信息📚第二章 安装📗下载源码📗下载安装包📗修改版本📗解压安装包📗运行buildDockerImage.sh脚本📗docker操作📕查看docker镜像📕启动dock…...

微信小程序 uniapp+vue城市公交线路查询系统dtjl3

小程序Android端运行软件 微信开发者工具/hbuiderx uni-app框架:使用Vue.js开发跨平台应用的前端框架,编写一套代码,可编译到Android、小程序等平台。 前端:HTML5,CSS3 VUE 后端:java(springbootssm)/python(flaskdja…...

2024年MathorCup数模竞赛B题问题一二三+部分代码分享

inputFolderPath E:\oracle\images\; outputFolderPath E:\oracle\process\; % 获取文件夹中所有图片的文件列表 imageFiles dir(fullfile(inputFolderPath, *.jpg)); % 设置colorbar范围阈值 threshold 120; % 遍历每个图片文件 for i 1:length(imageFiles) % 读…...

Ubuntu日常配置

目录 修改网络配置 xshell连不上怎么办 解析域名失败 永久修改DNS方法 临时修改DNS方法 修改网络配置 1、先ifconfig确认本机IP地址(刚装的机子没有ifconfig,先apt install net-tools) 2、22.04版本的ubuntu网络配置在netplan目录下&…...

GMSSL-通信

死磕GMSSL通信-C/C++系列(一) 最近再做国密通信的项目开发,以为国密也就简单的集成一个库就可以完事了,没想到能有这么多坑。遂写下文章,避免重复踩坑。以下国密通信的坑有以下场景 1、使用GMSSL guanzhi/GmSSL进行通信 2、使用加密套件SM2-WITH-SMS4-SM3 使用心得 ​…...

linux 磁盘分区Inode使用率达到100%,导致网站无法创建文件报错 failed:No space leftondevice(

linux 磁盘分区Inode使用率达到100%,导致网站无法创建文件报错 failed:No space left on device 由于这问题直接导致了,网站无法正常运行! 提交工单求助阿里后,得到了答案! 工程师先让我执行 df -h 和 df -i 通过分析…...

探索Python库的奇妙世界

探索Python库的奇妙世界 Python作为一种流行的编程语言,因其简洁的语法、强大的库支持和广泛的应用场景而备受开发者青睐。在这篇文章中,我们将深入探讨Python库的世界,了解它们如何帮助我们更高效地编写代码,并展示一些最有用的…...

SQL Server 存储函数(funGetId):唯一ID

系统测试时批量生成模拟数据,通过存储函数生成唯一ID。 根据当前时间生成唯一ID(17位) --自定义函数:根据当前时间组合成一个唯一ID字符串:yearmonthdayhourminutesecondmillisecond drop function funGetId;go--自定义函数&…...

当你的项目体积比较大?你如何做性能优化

在前端开发中,项目体积优化是一个重要的环节,它直接影响到网页的加载速度和用户体验。随着前端项目越来越复杂,引入的依赖也越来越多,如何有效地减少最终打包文件的大小,成为了前端工程师需要面对的挑战。以下是一些常…...

第6章:6.3.2 一张表总结正则表达式的语法 (MATLAB入门课程)

讲解视频:可以在bilibili搜索《MATLAB教程新手入门篇——数学建模清风主讲》。​ MATLAB教程新手入门篇(数学建模清风主讲,适合零基础同学观看)_哔哩哔哩_bilibili 本节我们用一张表来回顾和总结MATLAB正则表达式的基本语法。这个…...

VBA 实现outlook 当邮件设置category: red 即触发自动创建jira issue

1. 打开: Outlook VBA(Visual Basic for Applications) 方法一: 在邮件直接搜索:Visual Basic editor 方法二: File -> Options -> Customize Ribbon-> 打钩 如下图: 2.设置运行VBA 脚本: File -> Options -> Trust center -> Trus…...

进程地址空间(比特课总结)

一、进程地址空间 1. 环境变量 1 )⽤户级环境变量与系统级环境变量 全局属性:环境变量具有全局属性,会被⼦进程继承。例如当bash启动⼦进程时,环 境变量会⾃动传递给⼦进程。 本地变量限制:本地变量只在当前进程(ba…...

【WiFi帧结构】

文章目录 帧结构MAC头部管理帧 帧结构 Wi-Fi的帧分为三部分组成:MAC头部frame bodyFCS,其中MAC是固定格式的,frame body是可变长度。 MAC头部有frame control,duration,address1,address2,addre…...

MFC内存泄露

1、泄露代码示例 void X::SetApplicationBtn() {CMFCRibbonApplicationButton* pBtn GetApplicationButton();// 获取 Ribbon Bar 指针// 创建自定义按钮CCustomRibbonAppButton* pCustomButton new CCustomRibbonAppButton();pCustomButton->SetImage(IDB_BITMAP_Jdp26)…...

【JVM】- 内存结构

引言 JVM:Java Virtual Machine 定义:Java虚拟机,Java二进制字节码的运行环境好处: 一次编写,到处运行自动内存管理,垃圾回收的功能数组下标越界检查(会抛异常,不会覆盖到其他代码…...

Element Plus 表单(el-form)中关于正整数输入的校验规则

目录 1 单个正整数输入1.1 模板1.2 校验规则 2 两个正整数输入&#xff08;联动&#xff09;2.1 模板2.2 校验规则2.3 CSS 1 单个正整数输入 1.1 模板 <el-formref"formRef":model"formData":rules"formRules"label-width"150px"…...

Netty从入门到进阶(二)

二、Netty入门 1. 概述 1.1 Netty是什么 Netty is an asynchronous event-driven network application framework for rapid development of maintainable high performance protocol servers & clients. Netty是一个异步的、基于事件驱动的网络应用框架&#xff0c;用于…...

苹果AI眼镜:从“工具”到“社交姿态”的范式革命——重新定义AI交互入口的未来机会

在2025年的AI硬件浪潮中,苹果AI眼镜(Apple Glasses)正在引发一场关于“人机交互形态”的深度思考。它并非简单地替代AirPods或Apple Watch,而是开辟了一个全新的、日常可接受的AI入口。其核心价值不在于功能的堆叠,而在于如何通过形态设计打破社交壁垒,成为用户“全天佩戴…...

学习一下用鸿蒙​​DevEco Studio HarmonyOS5实现百度地图

在鸿蒙&#xff08;HarmonyOS5&#xff09;中集成百度地图&#xff0c;可以通过以下步骤和技术方案实现。结合鸿蒙的分布式能力和百度地图的API&#xff0c;可以构建跨设备的定位、导航和地图展示功能。 ​​1. 鸿蒙环境准备​​ ​​开发工具​​&#xff1a;下载安装 ​​De…...

Python 高效图像帧提取与视频编码:实战指南

Python 高效图像帧提取与视频编码:实战指南 在音视频处理领域,图像帧提取与视频编码是基础但极具挑战性的任务。Python 结合强大的第三方库(如 OpenCV、FFmpeg、PyAV),可以高效处理视频流,实现快速帧提取、压缩编码等关键功能。本文将深入介绍如何优化这些流程,提高处理…...

【Post-process】【VBA】ETABS VBA FrameObj.GetNameList and write to EXCEL

ETABS API实战:导出框架元素数据到Excel 在结构工程师的日常工作中,经常需要从ETABS模型中提取框架元素信息进行后续分析。手动复制粘贴不仅耗时,还容易出错。今天我们来用简单的VBA代码实现自动化导出。 🎯 我们要实现什么? 一键点击,就能将ETABS中所有框架元素的基…...