设计模式-中介者(调停者)模式(行为型)

news2024/12/26 14:07:17

中介者模式

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

图解

请添加图片描述

角色

  1. 同事对象:定义抽象接口,用于与中介者进行通信,一般是一个发送消息的接口,一个接收消息的接口
  2. 具体同事对象:实现同事对象抽象接口,具体发送、接收消息的逻辑
  3. 中介者:定义要给抽象接口用于与同事对象进行通信
  4. 具体中介者:实现同事之间通信的逻辑

案例1-中介者持有同事对象的引用

中介者

public interface Mediator {
    void execute();
}
/** 实现1*/
public class AllPersonMediator implements Mediator {
    private List<Person> personList;
    public AllPersonMediator(List<Person> personList) {
        this.personList = personList;
    }
    @Override
    public 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;
    }
    @Override
    public 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;
    }
    @Override
    public 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{
    @Override
    public void add(Client c) {
        queue.add(c);
    }
    @Override
    public 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);
    }

    @Override
    void sent() {
        new RuntimeException("这是接收消息的客户端");
    }

    @Override
    void 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);
    }

    @Override
    void sent() {
        simpleMediator.add(this);
        System.out.println("消息发送成功!");    }


    @Override
    void 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();
        }
    }
}

总结

在学习中介者模式的时候一直疑惑,为什么要有中介者接口?

学习完后感悟:设计模式是一种思想,不应该去套公式。中介者模式的核心思想就是通过对象引用的方式实现多个同事类通过一个中介者建立联系,可以中介者中引用同事类,也可以同时类引用中介者,甚至可以用一个不用中介者接口,直接定义具体中介者实现。具体怎么实现要根据具体情况而论。

设计模式是一种思想,而不是一种公式模板。

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

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

相关文章

湖南源点(市场研究咨询)如何产出更加有意义的竞品调研

湖南源点咨询认为&#xff1a;当前&#xff0c;任何项目都不能盲目开始&#xff0c;前期的准备工作必不可少。在基础架构搭建的同时&#xff0c;设计上对于前端功能、用户体验的调研就优先开始了。在这个阶段&#xff0c;大部分设计师都会分配很多调研任务&#xff0c;疯狂对竞…

BC C language

题目汇总 No.1 打印有规律的字符(牛牛的字符菱形) 代码展示 #include<stdio.h> int main() {char ch0;scanf("%c",&ch);for(int i0;i<5;i){for(int j0;j<5;j){if((i0||i4)&&j2)printf("%c", ch);else if ((i 1||i3) &&…

大疆智图_空三二维重建成果传输

一、软件环境 1.1 所需软件 1、 大疆智图&#xff1a;点击下载&#xff1b;   2、 ArcGIS Pro 3.1.5&#xff1a;点击下载&#xff0c;建议使用IDM或Aria2等多线程下载器&#xff1b;   3、 IDM下载器&#xff1a;点击下载&#xff0c;或自行搜索&#xff1b;   4、 Fas…

最近一直没动静的Pika Labs原来在筹集融资,加快构建视频基础模型

Pika 筹集了 8000 万美元&#xff0c;因此任何人都可以根据命令制作视频。 今天对我们来说是一个重要的日子。自从我们从斯坦福大学退学去构建 Pika 以来已经一年了&#xff0c;在这段时间里&#xff0c;我们在 Discord 上进行了秘密发布&#xff0c;发布了我们的 1.0 模型和 …

找了半天,还不如自己写一个图片转ico格式的程序

关于jpg、png等图片转ICO格式 最近突然急需一张ico格式的文件&#xff0c;就拿着处理好的png图片出网上找在线转换器&#xff0c;找了一个小时&#xff0c;绝了&#xff0c;不是需要注册充钱就是下载不下来&#xff0c;好不容易下载下来还是个文件错误。想着找个PS插件直接导出…

2024 cicsn ezbuf

文章目录 参考protobuf逆向学习复原结构思路exp 参考 https://www.y4ng.cn/posts/pwn/protobuf/#ciscn-2024-ezbuf protobuf 当时压根不知道用了protobuf这个玩意&#xff0c;提取工具也没提取出来&#xff0c;还是做题做太少了&#xff0c;很多关键性的结构都没看出来是pro…

WEB漏洞服务能提供哪些帮助

在数字化浪潮的推动下&#xff0c;Web应用程序已成为企业展示形象、提供服务、与用户进行交互的重要平台。然而&#xff0c;随着技术的飞速发展&#xff0c;Web应用程序中的安全漏洞也日益显现&#xff0c;成为网络安全的重大隐患。这些漏洞一旦被恶意攻击者利用&#xff0c;可…

极简主义在UI设计中的应用及解析

极简主义&#xff0c;即“少就是多”。在设计中&#xff0c;极简主义是许多艺术概念之一&#xff0c;它描述了一种内容形式&#xff0c;可以在许多方面使用。现在移动UI界面和网页设计中的极简主义设计越来越多。即时设计认为&#xff0c;极简主义UI界面不仅美观&#xff0c;而…

The 18th Northeast Collegiate Programming Contest(5/9/13)

心得 赛中ac&#xff1a;5&#xff0c;目前ac&#xff1a;9&#xff0c;题目总数&#xff1a;13 中档可做题还是很多的&#xff0c;可惜遇到了难绷的queueforces&#xff0c; 最后15min才判出来&#xff0c;oi赛制5wa4遗憾离场&#xff0c;赛后把几个题都给调过了&#xff0…

mysql (事物)

一.什么是事物 事物是一组操作的集合&#xff0c;不可分割的工作单位&#xff0c;事物会把所有的操作当作一个整体一起向系统提交或撤销操作请求&#xff0c;就是这些操作要么一起成功要么一起失败。 二.事物操作 &#xff08;这个就是一个理解&#xff09; 1.事务特性 原子性…

MongoDB CRUD操作:可重试写入

MongoDB CRUD操作&#xff1a;可重试写入 文章目录 MongoDB CRUD操作&#xff1a;可重试写入使用的先决条件部署的限制支持的存储引擎3.6 MongoDB 驱动程序MongoDB 版本写确认 可重试写入和多文档事务启用可重试写入MongoDB驱动mongosh 可重试的写操作行为持续的网络错误故障切…

46-1 护网溯源 - 钓鱼邮件溯源

一、客户提供钓鱼邮件样本 二、行为分析 三、样本分析 对钓鱼邮件中的木马程序1111.exe文件进行了分析,提交了360安全大脑沙箱云和微步在线云沙箱。 360安全大脑沙箱云显示,该1111.exe文件存在危险,因此在解压时需要谨慎操作,以免触发木马程序。 建议使用360压缩软件进行…

Chrome DevTools

Console 面板 此章节请打开 justwe7.github.io/devtools/console/console.html 一起食用 一方面用来记录页面在执行过程中的信息&#xff08;一般通过各种 console 语句来实现&#xff09;&#xff0c;另一方面用来当做 shell 窗口来执行脚本以及与页面文档、DevTools 等进行交…

Linux云计算架构师涨薪班就业服务有哪些?

学员一站式就业服务:一次学习&#xff0c;薪资翻倍 简历制作与指导 学员在培训期间&#xff0c;人才顾问会提供简历制作和指导服务&#xff0c;帮助学员制作出一份专业、有吸引力的简历。简历是求职者给招聘单位的第一印象&#xff0c;因此非常重要 模拟面试与技巧指导 为了让…

一种方法实现latex公式中显示空格

要解决的问题 我想要实现latex中出现空格。比如打出 Dice coefficients,然而一般情况下是不会显示出这两个单词之间的空格的&#xff0c;对吧。 解决方法 使用 \verb| |。 D i c e c o e f f i c i e n t s Dice coefficients Dicecoefficients D i c e c o e f f i c i e …

还不会线程池?JUC线程池源码级万字解析

线程池主要解决了两个问题&#xff1a; 第一个是当大量执行异步任务的时候提供较好的性能&#xff1b;在不使用线程池的时候&#xff0c;每次需要执行一个异步任务都需要新建一个 Thread 来进行&#xff0c;而线程的创建和销毁都是需要时间的&#xff0c;所以可以通过线程池来实…

数据集笔记:DGraph 大规模动态图数据集

dgraph-web (xinye.com) 1 数据集介绍 DGraph 是一个有向无权的动态图&#xff0c;包含超过 370 万个节点以及 430 万条动态边DGraph 中的节点表示金融借贷用户&#xff0c;有向边表示紧急联系人关系&#xff0c;每个节点包含脱敏后的属性特征&#xff0c;以及表示是否为金融…

Java 的循环

Java 有三种循环&#xff1a;for&#xff0c;while&#xff0c;do while。 for 基本语法&#xff1a; for (循环变量初始化; 循环条件; 循环变量迭代){循环语句; }程序示例&#xff1a; public static void main(String[] args) {for (int i 0, j 0; i < 3; i, j--) {…

45-2 waf绕过 - XSS 绕过WAF方法

环境准备: 43-5 waf绕过 - 安全狗简介及安装-CSDN博客然后安装pikachu靶场:构建完善的安全渗透测试环境:推荐工具、资源和下载链接_渗透测试靶机下载-CSDN博客打开pikachu靶场 http://127.0.0.1/pikachu-master/vul/xss/xss_reflected_get.php 使用常见payload被安全狗拦截…

树莓派 AI 套件,售价 70 美元

系列文章目录 前言 2024 年 6 月 4 日 Naush Patuck 如果您曾想在您的 Raspberry Pi 5 上尝试神经网络、人工智能和机器学习&#xff0c;我们为您准备了完美的产品&#xff1a;Raspberry Pi AI Kit。AI Kit 是与 Hailo 合作开发的&#xff0c;它提供了一种便捷的方法&…