《JAVA与模式》之调停者模式

系列文章目录


文章目录

  • 系列文章目录
  • 前言
  • 一、为什么需要调停者
  • 二、调停者模式的结构
  • 三、使用电脑来看电影


前言

前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站,这篇文章男女通用,看懂了就去分享给你的码吧。
在这里插入图片描述
在阎宏博士的《JAVA与模式》一书中开头是这样描述调停者(Mediator)模式的:

调停者模式是对象的行为模式。调停者模式包装了一系列对象相互作用的方式,使得这些对象不必相互明显引用。从而使它们可以较松散地耦合。当这些对象中的某些对象之间的相互作用发生改变时,不会立即影响到其他的一些对象之间的相互作用。从而保证这些相互作用可以彼此独立地变化。


一、为什么需要调停者

如下图所示,这个示意图中有大量的对象,这些对象既会影响别的对象,又会被别的对象所影响,因此常常叫做同事(Colleague)对象。这些同事对象通过彼此的相互作用形成系统的行为。从图中可以看出,几乎每一个对象都需要与其他的对象发生相互作用,而这种相互作用表现为一个对象与另一个对象的直接耦合。这就是过度耦合的系统。
在这里插入图片描述
通过引入调停者对象(Mediator),可以将系统的网状结构变成以中介者为中心的星形结构,如下图所示。在这个星形结构中,同事对象不再通过直接的联系与另一个对象发生相互作用;相反的,它通过调停者对象与另一个对象发生相互作用。调停者对象的存在保证了对象结构上的稳定,也就是说,系统的结构不会因为新对象的引入造成大量的修改工作。
在这里插入图片描述
一个好的面向对象的设计可以使对象之间增加协作性(Collaboration),减少耦合度(Couping)。一个深思熟虑的设计会把一个系统分解为一群相互协作的同事对象,然后给每一个同事对象以独特的责任,恰当的配置它们之间的协作关系,使它们可以在一起工作。

如果没有主板
  大家都知道,电脑里面各个配件之间的交互,主要是通过主板来完成的。如果电脑里面没有了主板,那么各个配件之间就必须自行相互交互,以互相传送数据。而且由于各个配件的接口不同,相互之间交互时,还必须把数据接口进行转换才能匹配上。

在这里插入图片描述
  所幸是有了主板,各个配件的交互完全通过主板来完成,每个配件都只需要和主板交互,而主板知道如何跟所有的配件打交道,这样就简单多了。
在这里插入图片描述

二、调停者模式的结构

调停者模式的示意性类图如下所示:
在这里插入图片描述
调停者模式包括以下角色:

●  抽象调停者(Mediator)角色:定义出同事对象到调停者对象的接口,其中主要方法是一个(或多个)事件方法。

●  具体调停者(ConcreteMediator)角色:实现了抽象调停者所声明的事件方法。具体调停者知晓所有的具体同事类,并负责具体的协调各同事对象的交互关系。

●  抽象同事类(Colleague)角色:定义出调停者到同事对象的接口。同事对象只知道调停者而不知道其余的同事对象。

●  具体同事类(ConcreteColleague)角色:所有的具体同事类均从抽象同事类继承而来。实现自己的业务,在需要与其他同事通信的时候,就与持有的调停者通信,调停者会负责与其他的同事交互。

源代码
  抽象调停者类

public interface Mediator {/*** 同事对象在自身改变的时候来通知调停者方法* 让调停者去负责相应的与其他同事对象的交互*/public void changed(Colleague c);
}

具体调停者类

public class ConcreteMediator implements Mediator {//持有并维护同事Aprivate ConcreteColleagueA colleagueA;//持有并维护同事Bprivate ConcreteColleagueB colleagueB;    public void setColleagueA(ConcreteColleagueA colleagueA) {this.colleagueA = colleagueA;}public void setColleagueB(ConcreteColleagueB colleagueB) {this.colleagueB = colleagueB;}@Overridepublic void changed(Colleague c) {/*** 某一个同事类发生了变化,通常需要与其他同事交互* 具体协调相应的同事对象来实现协作行为*/}}

抽象同事类

public abstract class Colleague {//持有一个调停者对象private Mediator mediator;/*** 构造函数*/public Colleague(Mediator mediator){this.mediator = mediator;}/*** 获取当前同事类对应的调停者对象*/public Mediator getMediator() {return mediator;}}

具体同事类

public class ConcreteColleagueA extends Colleague {public ConcreteColleagueA(Mediator mediator) {super(mediator);}/*** 示意方法,执行某些操作*/public void operation(){//在需要跟其他同事通信的时候,通知调停者对象getMediator().changed(this);}
}
public class ConcreteColleagueB extends Colleague {public ConcreteColleagueB(Mediator mediator) {super(mediator);}/*** 示意方法,执行某些操作*/public void operation(){//在需要跟其他同事通信的时候,通知调停者对象getMediator().changed(this);}
}

三、使用电脑来看电影

在日常生活中,我们经常使用电脑来看电影,把这个过程描述出来,简化后假定会有如下的交互过程:

(1)首先是光驱要读取光盘上的数据,然后告诉主板,它的状态改变了。

(2)主板去得到光驱的数据,把这些数据交给CPU进行分析处理。

(3)CPU处理完后,把数据分成了视频数据和音频数据,通知主板,它处理完了。

(4)主板去得到CPU处理过后的数据,分别把数据交给显卡和声卡,去显示出视频和发出声音。

要使用调停者模式来实现示例,那就要区分出同事对象和调停者对象。很明显,主板是调停者,而光驱、声卡、CPU、显卡等配件,都是作为同事对象。
  在这里插入图片描述
源代码
  抽象同事类

复制代码
public abstract class Colleague {
//持有一个调停者对象
private Mediator mediator;
/**
* 构造函数
/
public Colleague(Mediator mediator){
this.mediator = mediator;
}
/
*
* 获取当前同事类对应的调停者对象
*/
public Mediator getMediator() {
return mediator;
}
}
复制代码
  同事类——光驱

复制代码
public class CDDriver extends Colleague{
//光驱读取出来的数据
private String data = “”;
/**
* 构造函数
/
public CDDriver(Mediator mediator) {
super(mediator);
}
/
*
* 获取光盘读取出来的数据
/
public String getData() {
return data;
}
/
*
* 读取光盘
*/
public void readCD(){
//逗号前是视频显示的数据,逗号后是声音
this.data = “One Piece,海贼王我当定了”;
//通知主板,自己的状态发生了改变
getMediator().changed(this);
}
}
复制代码
  同事类——CPU

复制代码
public class CPU extends Colleague {
//分解出来的视频数据
private String videoData = “”;
//分解出来的声音数据
private String soundData = “”;
/**
* 构造函数
/
public CPU(Mediator mediator) {
super(mediator);
}
/
*
* 获取分解出来的视频数据
/
public String getVideoData() {
return videoData;
}
/
*
* 获取分解出来的声音数据
/
public String getSoundData() {
return soundData;
}
/
*
* 处理数据,把数据分成音频和视频的数据
*/
public void executeData(String data){
//把数据分解开,前面是视频数据,后面是音频数据
String[] array = data.split(“,”);
this.videoData = array[0];
this.soundData = array[1];
//通知主板,CPU完成工作
getMediator().changed(this);
}

}
复制代码
  同事类——显卡

复制代码
public class VideoCard extends Colleague {
/**
* 构造函数
/
public VideoCard(Mediator mediator) {
super(mediator);
}
/
*
* 显示视频数据
*/
public void showData(String data){
System.out.println(“您正在观看的是:” + data);
}
}
复制代码
  同事类——声卡

复制代码
public class SoundCard extends Colleague {
/**
* 构造函数
/
public SoundCard(Mediator mediator) {
super(mediator);
}
/
*
* 按照声频数据发出声音
*/
public void soundData(String data){
System.out.println(“画外音:” + data);
}
}
复制代码
  抽象调停者类

复制代码
public interface Mediator {
/**
* 同事对象在自身改变的时候来通知调停者方法
* 让调停者去负责相应的与其他同事对象的交互
*/
public void changed(Colleague c);
}
复制代码
  具体调停者类

复制代码
public class MainBoard implements Mediator {
//需要知道要交互的同事类——光驱类
private CDDriver cdDriver = null;
//需要知道要交互的同事类——CPU类
private CPU cpu = null;
//需要知道要交互的同事类——显卡类
private VideoCard videoCard = null;
//需要知道要交互的同事类——声卡类
private SoundCard soundCard = null;

public void setCdDriver(CDDriver cdDriver) {this.cdDriver = cdDriver;
}public void setCpu(CPU cpu) {this.cpu = cpu;
}public void setVideoCard(VideoCard videoCard) {this.videoCard = videoCard;
}public void setSoundCard(SoundCard soundCard) {this.soundCard = soundCard;
}@Override
public void changed(Colleague c) {if(c instanceof CDDriver){//表示光驱读取数据了this.opeCDDriverReadData((CDDriver)c);}else if(c instanceof CPU){this.opeCPU((CPU)c);}
}
/*** 处理光驱读取数据以后与其他对象的交互*/
private void opeCDDriverReadData(CDDriver cd){//先获取光驱读取的数据String data = cd.getData();//把这些数据传递给CPU进行处理cpu.executeData(data);
}
/*** 处理CPU处理完数据后与其他对象的交互*/
private void opeCPU(CPU cpu){//先获取CPU处理后的数据String videoData = cpu.getVideoData();String soundData = cpu.getSoundData();//把这些数据传递给显卡和声卡展示出来videoCard.showData(videoData);soundCard.soundData(soundData);
}

}
复制代码
  客户端类

复制代码
public class Client {

public static void main(String[] args) {//创建调停者——主板MainBoard mediator = new MainBoard();//创建同事类CDDriver cd = new CDDriver(mediator);CPU cpu = new CPU(mediator);VideoCard vc = new VideoCard(mediator);SoundCard sc = new SoundCard(mediator);//让调停者知道所有同事mediator.setCdDriver(cd);mediator.setCpu(cpu);mediator.setVideoCard(vc);mediator.setSoundCard(sc);//开始看电影,把光盘放入光驱,光驱开始读盘cd.readCD();}

}
在这里插入图片描述
调停者模式的优点
  ●  松散耦合

调停者模式通过把多个同事对象之间的交互封装到调停者对象里面,从而使得同事对象之间松散耦合,基本上可以做到互补依赖。这样一来,同事对象就可以独立地变化和复用,而不再像以前那样“牵一处而动全身”了。

●  集中控制交互

多个同事对象的交互,被封装在调停者对象里面集中管理,使得这些交互行为发生变化的时候,只需要修改调停者对象就可以了,当然如果是已经做好的系统,那么就扩展调停者对象,而各个同事类不需要做修改。

●  多对多变成一对多

没有使用调停者模式的时候,同事对象之间的关系通常是多对多的,引入调停者对象以后,调停者对象和同事对象的关系通常变成双向的一对多,这会让对象的关系更容易理解和实现。

调停者模式的缺点
  调停者模式的一个潜在缺点是,过度集中化。如果同事对象的交互非常多,而且比较复杂,当这些复杂性全部集中到调停者的时候,会导致调停者对象变得十分复杂,而且难于管理和维护。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/508873.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【二分查找】【C++算法】378. 有序矩阵中第 K 小的元素

作者推荐 视频算法专题 本文涉及的基础知识点 二分查找算法合集 LeetCode378. 有序矩阵中第 K 小的元素 给你一个 n x n 矩阵 matrix ,其中每行和每列元素均按升序排序,找到矩阵中第 k 小的元素。 请注意,它是 排序后 的第 k 小元素&…

深度学习_15_过拟合欠拟合

过拟合和欠拟合 过拟合和欠拟合是训练模型中常会发生的事,如所要识别手势过于复杂,如五角星手势,那就需要更改高级更复杂的模型去训练,若用比较简单模型去训练,就会导致模型未能抓住手势的全部特征,那简单…

了解游戏中的数据同步

目录 数据同步 通过比较来看状态同步和帧同步 状态同步 帧同步 帧同步实现需要的条件 两者相比较 数据同步 在联机游戏中,我的操作和数据要同步给同一局游戏中其他所有玩家,其他玩家的操作和数据也会同步给我。这叫做数据同步,目前数据…

《TCP/IP详解 卷一》第10章 UDP 和 IP 分片

目录 10.1 引言 10.2 UDP 头部 10.3 UDP校验和 10.4 例子 10.5 UDP 和 IPv6 10.6 UDP-Lite 10.7 IP分片 10.7.1 例子:IPV4 UDP分片 10.7.2 重组超时 10.8 采用UDP的路径MTU发现 10.9 IP分片和ARP/ND之间的交互 10.10 最大UDP数据报长度 10.11 UDP服务器…

代码随想录Day66 | 图的DFS与BFS

代码随想录Day66 | 图的DFS与BFS DFS797.所有可能的路径无向图和有向图的处理 BFS200.岛屿数量 DFS 文档讲解:代码随想录 视频讲解: 状态 本质上就是回溯算法。 void dfs(参数) {if (终止条件) {存放结果;return;}for (选择:本节点所连接的…

猴子吃桃问题(python版)

文章预览: 题目python解法一:运行结果 python解法二:运行结果 python解法三:运行结果 题目 猴子吃桃问题:猴子第一天摘下若干个桃子,当即吃了一半,还不过瘾,又多吃了一个。 第二天早…

超全面!Linux学习资料大合集,21套从入门到进阶,看这篇就够了

本文将为那些渴望学习Linux,但又缺乏相应资料和方向的朋友,提供21套Linux优质资料,包含入门到进阶,希望能对大家有所帮助。 此合集内容及其丰富,涉及方面颇多,不仅适合Linux入门学习的朋友,运维…

水电站数字孪生:水力发电在可视化领域的应用

自水轮机的早期发明被用于农业灌溉,到 18 世纪末期的工业革命促使水轮机技术的改良,再到 19 世纪末水利发电的崛起,直至今日,智慧水电站数字孪生技术正处于蓬勃发展之中。通过整合物联网、大数据、云计算等现代信息技术&#xff0…

【Zookeeper】ZooKeeper的一些重要功能和作用

🍎个人博客:个人主页 🏆个人专栏:日常聊聊 ⛳️ 功不唐捐,玉汝于成 目录 前言 正文 结语 我的其他博客 前言 随着分布式系统的普及和应用场景的不断增加,构建可靠、高效的分布式系统变得愈发重要。然…

开短路测试

一、介绍 连接性测试,可以叫开短路测试,也可叫接触性测试,英文为Continuity Test或Open & Short Test,主要用来检验测试过程中电学连接是否良好,包括测试设备本身、测试设备与Loadboard、DUT本身等等,是…

SpringBoot接口防抖(防重复提交)的一些实现方案

前言 啥是防抖 思路解析 分布式部署下如何做接口防抖? 具体实现 请求锁 唯一key生成 重复提交判断 前言 作为一名老码农,在开发后端Java业务系统,包括各种管理后台和小程序等。在这些项目中,我设计过单/多租户体系系统&a…

一文梳理LIN协议与应用

目录 一、LIN是什么?有什么用?二、LIN概要描述2.1 特点2.1.1 低成本2.1.2 易扩展2.1.3 速率一般,不高 2.2 网络形态2.3 架构分层 三、应用层 4 大功能3.1 配置3.2 识别(查询功能)3.3 信号处理3.4 诊断 四、协议层介绍4…