设计模式-调停者模式

news2025/1/20 20:04:07
设计模式专栏

    • 模式介绍
    • 模式特点
    • 应用场景
    • 调停者模式与命令模式的比较
    • 代码示例
      • Java实现调停者模式
      • Python实现调停者模式
    • 调停者模式在spring中的应用


模式介绍

调停者模式是一种软件设计模式,主要用于模块间的解耦,通过避免对象之间显式的互相指向,降低耦合度。这种模式通常用于解决多个对象之间重复访问、无规律性、交互方式复杂且通常存在非结构依赖的问题。

在调停者模式中,通常会有一个抽象调停者(Mediator)类,它定义了同事对象与调停者对象之间的接口,主要方法是一个或多个事件方法。具体调停者(ConcreteMediator)类则实现了抽象调停者所声明的事件方法,负责具体的协调各同事对象的交互关系。此外,还有一个抽象同事类(Colleague),它定义了调停者到同事对象的接口,同事对象只知道调停者而不知道其余的同事对象。

调停者模式将多对多的相互作用转化为一对多的相互作用,将对象的行为和协作抽象化,把对象在小尺度的行为上与其他对象的相互作用分开处理。通过使用中介对象封装一系列的对象交互,各对象不需要显式地相互引用,从而使其耦合松散,而且可以独立地改变它们之间的交互。这种模式有助于降低系统的复杂性,提高可维护性和可扩展性。

在这里插入图片描述

模式特点

调停者模式的特点主要包括以下几个方面:

  1. 封装交互 :调停者模式将一系列对象之间的交互封装在一个单独的调停者对象中,使得这些对象之间不需要相互明显引用,从而降低了耦合度。
  2. 抽象化 :调停者模式将对象的行为和协作抽象化,将小尺度的行为上与其他对象的相互作用分开处理,使得对象之间的关系更加易于维护和理解。
  3. 转化交互方式 :调停者模式将多对多的相互作用转化为一对多的相互作用,使得对象之间的关系更加清晰和易于管理。
  4. 提供可扩展性 :调停者模式为同事对象提供可扩展性,使得系统可以在不改变现有类的情况下进行扩展。
  5. 控制集中化 :调停者模式将控制集中在一个对象上,可能会使得这个对象变得复杂且难于管理。
  6. 适用场景 :调停者模式适用于解决多个对象之间存在复杂交互、行为变化多样且不规律的问题,尤其在涉及不同对象的协同工作、请求分发、业务逻辑协调等场景中能够发挥重要作用。

调停者模式通过封装交互、抽象化、转化交互方式等特点,降低了系统的耦合度,使得对象间的交互行为更加清晰和易于管理,提高了系统的可维护性和可扩展性。同时,需要注意控制集中化的问题,并合理选择适用场景。

在这里插入图片描述

应用场景

调停者模式的应用场景主要包括:

  1. 当一个系统内部存在许多类,它们之间相互调用以完成一系列功能时,一旦某个类发生问题或需要进行修改,这可能会导致对所有调用它的类产生影响,此时就可以采用调停者模式。通过引入调停者类来封装所有类之间的交互,使原本复杂的网状结构变得简单,形成星型结构,降低了类与类之间的耦合度。
  2. 在处理对象之间存在复杂交互、行为变化多样且不规律的情况下,调停者模式可以起到很好的作用。
  3. 存在非结构化依赖的情况下,使用调停者模式能够简化对象的相互依赖关系,降低维护和扩展的难度。
  4. 涉及不同对象的协同工作、请求分发、业务逻辑协调等场景,也可以考虑使用调停者模式。

调停者模式能够将对象间的直接调用解耦,使得对象间的交互行为集中化管理,方便维护和扩展。然而,这种模式也有其局限性,如可能导致控制集中化,使得调停者类变得复杂且难于管理。在实际应用中,需要根据具体问题来决定是否采用调停者模式。

在这里插入图片描述

调停者模式与命令模式的比较

调停者模式与命令模式是两种不同的软件设计模式,它们有不同的应用场景和特点。

调停者模式主要用于解决多个对象之间的交互问题,通过引入一个调停者对象来封装和协调各个对象之间的交互,降低系统的耦合度,使得对象间的交互行为更加清晰和易于管理。其主要特点是能够将多个对象的交互行为集中在一个对象中管理,使得对象的相互依赖关系变得简单,降低系统的复杂性和维护成本。

命令模式是一种行为型设计模式,它主要处理在不同对象之间指定责任的细节,描述对象间的通信机制,定义一种机制以便在运行时根据不同的对象选择不同的逻辑。其主要特点是将请求或操作封装到一个对象中,通过使用命令对象来请求一个操作的对象与知道怎么执行一个操作的对象解耦,使得请求可以被取消、恢复或者组合。

调停者模式与命令模式的不同点在于,调停者模式主要关注多个对象之间的交互解耦,而命令模式主要关注请求或操作的封装、排队和撤销等控制功能。此外,调停者模式通常用于协调对象之间的交互行为,而命令模式通常用于处理请求或操作的控制流程。

相同点方面,调停者模式和命令模式都关注降低系统的耦合度,提高系统的可维护性和可扩展性。它们都通过封装请求或操作来实现对请求的控制,包括排队、顺序执行和撤销等。同时,它们都需要在请求的发送者和接收者之间建立一种抽象的通信机制,以便在运行时根据不同的对象选择不同的逻辑。

调停者模式和命令模式在应用场景、作用和使用特点等方面存在明显的差异,但两者在抽象通信机制和降低系统耦合度方面有一定的相似性。在实际应用中,根据具体的问题和需求选择合适的模式是重要的。

在这里插入图片描述

代码示例

Java实现调停者模式

以下是一个简单的Java实现调停者模式的示例:

// 抽象调停者
interface Mediator {
    void register(Colleague colleague);
    void handleMessage(Colleague colleague, Object message);
}

// 具体调停者
class ConcreteMediator implements Mediator {
    private Map<Colleague, Handler> handlers = new HashMap<>();

    @Override
    public void register(Colleague colleague) {
        handlers.put(colleague, new Handler(colleague));
    }

    @Override
    public void handleMessage(Colleague colleague, Object message) {
        Handler handler = handlers.get(colleague);
        if (handler != null) {
            handler.handleMessage(message);
        }
    }

    class Handler {
        private Colleague colleague;

        public Handler(Colleague colleague) {
            this.colleague = colleague;
        }

        public void handleMessage(Object message) {
            // 处理消息逻辑,根据具体需求实现
            System.out.println("ConcreteMediator handled message for Colleague " + colleague);
        }
    }
}

// 抽象同事类
interface Colleague {
    void setMediator(Mediator mediator);
    void sendMessage(Object message);
}

// 具体同事类1
class ConcreteColleague1 implements Colleague {
    private Mediator mediator;
    private String name;

    public ConcreteColleague1(String name) {
        this.name = name;
    }

    @Override
    public void setMediator(Mediator mediator) {
        this.mediator = mediator;
    }

    @Override
    public void sendMessage(Object message) {
        mediator.handleMessage(this, message);
    }

    @Override
    public String toString() {
        return name;
    }
}

// 具体同事类2
class ConcreteColleague2 implements Colleague {
    private Mediator mediator;
    private String name;

    public ConcreteColleague2(String name) {
        this.name = name;
    }

    @Override
    public void setMediator(Mediator mediator) {
        this.mediator = mediator;
    }

    @Override
    public void sendMessage(Object message) {
        mediator.handleMessage(this, message);
    }

    @Override
    public String toString() {
        return name;
    }
}

Python实现调停者模式

以下是Python实现调停者模式的示例代码:

from abc import ABC, abstractmethod

# 抽象调停者
class Mediator(ABC):
    @abstractmethod
    def register(self, colleague):
        pass

    @abstractmethod
    def handle_message(self, colleague, message):
        pass

# 具体调停者
class ConcreteMediator(Mediator):
    def __init__(self):
        self.colleagues = []

    def register(self, colleague):
        self.colleagues.append(colleague)

    def handle_message(self, colleague, message):
        for c in self.colleagues:
            if c != colleague:
                c.send_message(message)

# 抽象同事类
class Colleague(ABC):
    @abstractmethod
    def set_mediator(self, mediator):
        pass

    @abstractmethod
    def send_message(self, message):
        pass

# 具体同事类1
class ConcreteColleague1(Colleague):
    def __init__(self, name):
        self.name = name
        self.mediator = None

    def set_mediator(self, mediator):
        self.mediator = mediator
        mediator.register(self)

    def send_message(self, message):
        self.mediator.handle_message(self, message)
        print(f"{self.name} sent message: {message}")

# 具体同事类2
class ConcreteColleague2(Colleague):
    def __init__(self, name):
        self.name = name
        self.mediator = None

    def set_mediator(self, mediator):
        self.mediator = mediator
        mediator.register(self)

    def send_message(self, message):
        self.mediator.handle_message(self, message)
        print(f"{self.name} sent message: {message}")

在这里插入图片描述

调停者模式在spring中的应用

在Spring框架中,调停者模式可以应用于许多场景,其中最常见的是事件监听和通知机制。Spring的事件监听器模式就是一个典型的调停者模式的实现。

在Spring中,可以使用ApplicationListener接口和ApplicationEvent类来实现事件监听和通知。ApplicationListener接口定义了一个onApplicationEvent方法,该方法在接收到事件时会触发。ApplicationEvent类是事件的抽象类,它定义了一些常用的方法,如获取事件的来源和时间等。

具体实现时,首先需要创建一个继承自ApplicationEvent的事件类,例如:

public class MyCustomEvent extends ApplicationEvent {
    private String message;

    public MyCustomEvent(Object source, String message) {
        super(source);
        this.message = message;
    }

    public String getMessage() {
        return message;
    }
}

然后,创建一个实现ApplicationListener接口的监听器类,例如:

import org.springframework.context.ApplicationListener;
import org.springframework.stereotype.Component;

@Component
public class MyCustomEventListener implements ApplicationListener<MyCustomEvent> {
    @Override
    public void onApplicationEvent(MyCustomEvent event) {
        System.out.println("Received MyCustomEvent - " + event.getMessage());
    }
}

最后,在需要发布事件的类中,使用ApplicationContextpublishEvent方法发布事件:

import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import org.springframework.stereotype.Component;

@Component
public class EventPublisher {
    private ApplicationContext applicationContext;

    public EventPublisher(ApplicationContext applicationContext) {
        this.applicationContext = applicationContext;
    }

    public void publishEvent() {
        String message = "Hello, Spring!";
        MyCustomEvent event = new MyCustomEvent(this, message);
        applicationContext.publishEvent(event);
    }
}

在上面的示例中,EventPublisher组件发布了一个自定义的事件,所有实现了MyCustomEventListener接口的组件都会接收到该事件,并执行相应的逻辑。这就是Spring中调停者模式的典型应用。

在这里插入图片描述

设计模式-过滤器模式

设计模式-门面模式

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

YOLOv8改进 | 2023注意力篇 | iRMB倒置残差块注意力机制(轻量化注意力机制)

一、本文介绍 本文给家大家带来的改进机制是iRMB&#xff0c;其是在论文Rethinking Mobile Block for Efficient Attention-based Models种提出&#xff0c;论文提出了一个新的主干网络EMO(后面我也会教大家如何使用该主干&#xff0c;本文先教大家使用该文中提出的注意力机制…

Java超高精度无线定位技术--UWB (超宽带)人员定位系统源码

UWB室内定位技术是一种全新的、与传统通信技术有极大差异的通信新技术。它不需要使用传统通信体制中的载波&#xff0c;而是通过发送和接收具有纳秒或纳秒级以下的极窄脉冲来传输数据&#xff0c;从而具有GHz量级的带宽。 UWB&#xff08;超宽带&#xff09;高精度定位系统是一…

08.哲说建造者模式(Builder Pattern)

“The odds that we’re in ‘base reality’ is one in billions.” —— Elon Musk 这段话出自马斯克在2016年的一次演讲&#xff0c;“人类活在真实世界的几率&#xff0c;可能不到十亿分之一”。此言一出&#xff0c;可谓一石激起千层浪。有人嘲讽马斯克是“语不惊人死不休…

论文阅读——SG-Former

SG-Former: Self-guided Transformer with Evolving Token Reallocation 1. Introduction 方法的核心是利用显著性图&#xff0c;根据每个区域的显著性重新分配tokens。显著性图是通过混合规模的自我关注来估计的&#xff0c;并在训练过程中自我进化。直观地说&#xff0c;我们…

3D视觉-相机选用的原则

鉴于不同技术方案都有其适用的场景&#xff0c;立体相机的选型讲究的原则为“先看用途&#xff0c;再看场景&#xff0c;终评精度”&#xff0c;合适的立体相机在方案中可以起到事半功倍的效果。从用途上来进行划分&#xff0c;三维视觉方案主要应用在两个方向&#xff1a;测量…

PyTorch常用工具(1)数据处理

文章目录 前言1 数据处理1.1 Dataset1.2 DataLoader 前言 在训练神经网络的过程中需要用到很多的工具&#xff0c;最重要的是数据处理、可视化和GPU加速。本章主要介绍PyTorch在这些方面常用的工具模块&#xff0c;合理使用这些工具可以极大地提高编程效率。 由于内容较多&am…

【每日一题】一周中的第几天

文章目录 Tag题目来源解题思路方法一&#xff1a;模拟 写在最后 Tag 【模拟】【数学】【2023-12-30】 题目来源 1185. 一周中的第几天 解题思路 方法一&#xff1a;模拟 思路 题目中的日期是在 1971 到 2100 年之间的有效日期&#xff0c;即 1971-01-01 到 2100-12-31 范围…

Scene Creator

场景创建器是一个方便、易于使用的编辑工具&#xff0c;旨在简化创建新场景的过程。使用场景创建器&#xff0c;您可以选择一个模板场景&#xff0c;定义一个目录来存储您的场景&#xff0c;并在需要时自动将新场景添加到构建中。 下载&#xff1a; ​​Unity资源商店链接 资…

数据结构:第7章:查找(复习)

目录 顺序查找&#xff1a; 折半查找&#xff1a; 二叉排序树&#xff1a; 4. (程序题) 平衡二叉树&#xff1a; 顺序查找&#xff1a; ASL 折半查找&#xff1a; 这里 j 表示 二叉查找树的第 j 层 二叉排序树&#xff1a; 二叉排序树&#xff08;Binary Search Tree&…

李宏毅 自然语言处理(Voice Conversion) 笔记

前一章笔记&#xff1a;李宏毅 自然语言处理&#xff08;Speech Recognition&#xff09; 笔记 引入 什么是voice conversion&#xff1f; 输入一段声音&#xff0c;输出另一段声音&#xff0c;我们希望这两端声音&#xff1a;内容一样&#xff0c;其他方面不一样&#xff08…

6个火爆全网的AI开源项目,用上月10万+

标题月10万可能说的有点夸张和含糊&#xff0c;10万具体指的是你可以利用这些开源项目实现&#xff1a; 访问量10万 收入10万 用户10万 …… 开源项目只是免费的工具&#xff0c;具体怎么实现还需要你根据自己需求去深入运营。这里只是给你推荐一些比较热门的开源项目&…

html-css-js移动端导航栏底部固定+i18n国际化全局

需求&#xff1a;要做一个移动端的仿照小程序的导航栏页面操作&#xff0c;但是这边加上了i18n国家化&#xff0c;由于页面切换的时候会导致国际化失效&#xff0c;所以写了这篇文章 1.效果 切换页面的时候中英文也会跟着改变&#xff0c;不会导致切换后回到默认的语言 2.实现…

【ESP-NOW with ESP32:向多个开发板发送数据(一对多)】

【ESP-NOW with ESP32&#xff1a;向多个开发板发送数据&#xff08;一对多&#xff09;】 1. 项目概况2. 先决条件2.1 环境配置2.2 所需零件 3. 获取主板 MAC 地址4. ESP32 发射器 &#xff08;ESP-NOW&#xff09;4.1 接收方的MAC地址4.2 OnDataSent&#xff08;&#xff09;…

2023-12-16 LeetCode每日一题(统计区间中的整数数目)

2023-12-16每日一题 一、题目编号 2276. 统计区间中的整数数目二、题目链接 点击跳转到题目位置 三、题目描述 给你区间的 空 集&#xff0c;请你设计并实现满足要求的数据结构&#xff1a; **新增&#xff1a;**添加一个区间到这个区间集合中。 **统计&#xff1a;**计算…

Plantuml之甘特图语法介绍(二十八)

简介&#xff1a; CSDN博客专家&#xff0c;专注Android/Linux系统&#xff0c;分享多mic语音方案、音视频、编解码等技术&#xff0c;与大家一起成长&#xff01; 优质专栏&#xff1a;Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 优质专栏&#xff1a;多媒…

Transformer基本结构

Transformer基本结构 输入部分、编码部分、解码部分、输出部分 1、输入部分 原文本嵌入层及其位置编码器目标文本嵌入层及其位置编码器 位置编码器(PositionalEncoding)&#xff1a;将词汇位置不同可能会产生不同语义的信息加入到词张量中&#xff0c;以弥补位置信息的缺失 …

node版本管理器nvm的下载和使用

介绍 nvm 全名 node.js version management&#xff0c;顾名思义是一个nodejs的版本管理工具。通过它可以安装和切换不同版本的nodejs。 下载和安装 在下载和安装nvm前&#xff0c;需要确保当前电脑没有安装node&#xff0c;否则则需要先把原来的node卸载了。 下载地址&#…

HTML+CSS+JS制作三款雪花酷炫特效

🎀效果展示 🎀代码展示 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> <html

Fortran 编译后Dll依赖问题(libifcoremd.dll)

在VS里&#xff0c;点击运行按钮&#xff0c;程序正常运行。 在exe文件目录中&#xff0c;双击exe运行&#xff0c;提示缺少libifcoremd.dll文件&#xff0c;经查阅资料和以往的经验&#xff0c;解决方法如下&#xff1a; 红色是默认的&#xff0c;绿色是能够正常运行的。

AcWing算法提高课-2.3.1矩阵距离

算法提高课整理 CSDN个人主页&#xff1a;更好的阅读体验 本文同步发表于 CSDN | 洛谷 | AcWing | 个人博客 原题链接 题目描述 给定一个 01 矩阵&#xff0c;求矩阵中每个元素离 1 的最短曼哈顿距离。 输入格式 第一行两个整数 n , m n,m n,m。 接下来一个 n n n 行 …