zoukankan      html  css  js  c++  java
  • 【设计模式(17)】行为型模式之中介者模式

    个人学习笔记分享,当前能力有限,请勿贬低,菜鸟互学,大佬绕道

    如有勘误,欢迎指出和讨论,本文后期也会进行修正和补充


    前言

    相信很多人都对更换手机号有所顾虑,或者体验过了这种麻烦——通知每个人自己手机号变更了,亲友朋友同事上司全都得通知一遍,万一漏了,就失联了,甚至还得确保他们收到且更换了通讯录中你的号码

    那么,我们能不能将手机号存在一个中介处,我们只需要告知中介处我们的新手机号。当有人想找我们的时候,他们也只需要询问中介就行了。

    比如公司、班级、家庭的通讯录,自己手机号或者地址修改后,也同步修改一下就好了,找人的时候去上面查找即可。

    同样的例子,还有人才交流市场、聊天室、房屋中介等等,甚至是前台小姐姐都算是中介者~

    那么我们的网状关系结构,就变成了星型关系结构

    简而言之,其目的就是为了降低对象之间的耦合性,从而提高其灵活性


    1.介绍

    适用目的:定义一个中介对象来封装一系列对象之间的交互,使原有对象之间的耦合松散,且可以独立地改变它们之间的交互。

    主要解决:对象之间存在大量的关联关系,会导致系统结构复杂,若某一个对象发生改变,则需要跟踪处理所有有关的对象。

    何时使用:多个指责相似的类相互耦合,形成了网状结构。

    如何解决:对象 Colleague 之间的通信封装到一个类中统一处理。

    关键代码:定义中介者,用于存储和管理Colleague;定义Colleague持有中介者,并通过中介者相互通讯;

    应用实例:聊天室;MVC框架;资源调度系统;

    优点:

    • 降低了类的复杂度,将一对多转化成了一对一,提高了灵活性,易于扩展和维护
    • 各个类之间的解耦,降低了系统的耦合度
    • 类之间各司其职,符合迪米特原则

    缺点:Colleague越多,则中介者越臃肿,越难以维护

    使用场景

    • 系统中对象之间存在比较复杂的引用关系,导致它们之间的依赖关系结构混乱而且难以复用该对象
    • 想通过一个中间类来封装多个类中的行为,而又不想生成太多的子类。
    • 大量相似的类,相互之间若是网状关系则十分混乱

    注意事项:不应当在职责混乱的时候使用。


    2.结构

    中介者模式包含以下主要角色

    • 抽象中介者(Mediator)角色:它是中介者的接口,提供了同事对象注册与转发同事对象信息的抽象方法。
    • 具体中介者(Concrete Mediator)角色:实现中介者接口,定义一个 List 来管理同事对象,协调各个同事角色之间的交互关系,因此它依赖于同事角色。
    • 抽象同事类(Colleague)角色:定义同事类的接口,保存中介者对象,提供同事对象交互的抽象方法,实现所有相互影响的同事类的公共功能。
    • 具体同事类(Concrete Colleague)角色:是抽象同事类的实现者,当需要与其他同事对象交互时,由中介者对象负责后续的交互。

    image-20210224181126223


    3.实现步骤

    3.1.简单实例:模拟聊天室

    业务流程如下

    • 用户进入聊天室时进行注册
    • 用户发送消息时,会将消息转发给聊天室的其他用户
    • 用户收到消息时,显示消息内容和发送者

    代码如下

    1. 定义抽象中介者

      //抽象中介者
      abstract class Mediator {
          public abstract void register(Colleague colleague);
      
          public abstract void relay(Colleague colleague,String msg);
      }
      
    2. 定义具体中介者,实现抽象中介者的功能

      class ConcreteMediator extends Mediator {
      
          private Set<Colleague> colleagues = new HashSet<>();
      
          @Override
          public void register(Colleague colleague) {
              if (!colleagues.contains(colleague)) {
                  colleagues.add(colleague);
              }
          }
      
          @Override
          public void relay(Colleague colleague,String msg) {
              for (Colleague item : colleagues) {
                  if (!Objects.equals(item, colleague)) {
                      item.receive(colleague,msg);
                  }
              }
          }
      }
      

      此处使用Set作为容器,也可以使用ListMapQueue等容器,没啥区别

      转发的行为也可以是存入消息队列、命令池等,视情况而定

      请注意此处用对象比较,不要使用其内部的参数,避免两个不同的人使用相同标识符的情况,也给同事类留出更大的自定义扩展空间

    3. 定义抽象同事类

      abstract class Colleague {
      
          protected Mediator mediator;
          public String name;
      
          public Colleague(Mediator mediator, String name) {
              this.mediator = mediator;
              this.name = name;
          }
      
          public abstract void receive(Colleague colleague, String msg);
      
          public abstract void send(String msg);
      }
      

      此处的mediator即中介者,被同事持有,以便于向中介者传递消息,当然也可以使用单例模式或者向接口发送消息等解决方案

      此处的name为同事类的标识符,即用于分辨同事是谁,也可以使用uid等作为标识符

      请注意此处的标识符并不唯一,相当于生活中的姓名,是可能也允许出现重复的

    4. 定义具体同事类,实现抽象同事类

      class ConcreteColleague extends Colleague {
      
          public ConcreteColleague(Mediator mediator, String name) {
              super(mediator, name);
          }
      
          @Override
          public void receive(Colleague colleague,String msg) {
              System.out.println(name + " receive message from "+colleague.name+": " + msg);
          }
      
          @Override
          public void send(String msg) {
              System.out.println(name + " send message: " + msg);
              mediator.relay(this,msg);
          }
      }
      

      具体同事类可以定义多个,每个可以自行定义收发消息的行为,也可以存储自定义的信息

      例如:

      • 定义老师类,存储老师的联系方式、教师编号等信息

      • 定义学生类,存储学生的学号、所在班级等信息

      • 定义管理员类,存储联系方式即可

      但存在多种同事类的情况下,建议在抽象同事类中留有类型标记(如int type),以便于中介者识别

    测试代码

    public class Test {
        public static void main(String[] args) {
            Mediator mediator=new ConcreteMediator();
            Colleague colleagueA=new ConcreteColleague(mediator,"A");
            Colleague colleagueB=new ConcreteColleague(mediator,"B");
            Colleague colleagueC=new ConcreteColleague(mediator,"C");
            mediator.register(colleagueA);
            mediator.register(colleagueB);
            mediator.register(colleagueC);
    
    
            colleagueA.send("Hello! I'm A");
            System.out.println("-------------");
            colleagueB.send("Hello! I'm B");
            System.out.println("-------------");
            colleagueC.send("Hello! I'm C");
        }
    }
    

    运行结果

    image-20210225101254234

    3.2.进阶实例:模拟聊天室

    模拟业务

    • 建立一个聊天室,有三种类型的人员:学生、老师、GM
    • 每种身份均允许多个人参与
    • 每个人拥有一个独立的聊天框,显示所有人发送的消息

    全部代码

    package com.company.designPattern.mediator.test2;
    
    import javax.swing.*;
    import java.awt.*;
    import java.util.*;
    import java.awt.event.ActionEvent;
    import java.awt.event.ActionListener;
    import java.util.List;
    
    /**
     * @Description: 聊天室demo代码
     * @Author: Echo
     * @Time: 2021/2/25 10:22
     * @Email: 347110596@qq.com
     */
    
    public class Test {
        public static void main(String[] args) {
            //初始化聊天室
            Medium md = new ChatRoom();
            //初始化参与者
            Colleague member1 = new Student("小明");
            Colleague member2 = new Student("小红");
            Colleague member3 = new Teacher("张老师");
            Colleague member4 = new Manager("GM");
            //注册
            md.register(member1);
            md.register(member2);
            md.register(member3);
            md.register(member4);
        }
    }
    
    enum Types {
        Teacher("老师"),
        Student("学生"),
        Manager("管理员"),
        ;
    
        private final String TypeName;
    
        public String getTypeName() {
            return TypeName;
        }
    
        Types(String typeName) {
            TypeName = typeName;
        }
    }
    
    //抽象中介者
    interface Medium {
        void register(Colleague member); //注册
    
        void relay(Colleague from, String ad); //转发
    }
    
    //具体中介者:房地产中介
    class ChatRoom implements Medium {
        private List<Colleague> members = new ArrayList<>();
    
        public void register(Colleague member) {
            if (!members.contains(member)) {
                members.add(member);
                member.setMedium(this);
            }
        }
    
        public void relay(Colleague from, String msg) {
            for (Colleague to : members) {
                if (!Objects.equals(from, to)) {
                    to.receive(from, msg);
                }
            }
        }
    }
    
    //抽象同事类
    abstract class Colleague extends JFrame implements ActionListener {
        private static final long serialVersionUID = -7219939540794786080L;
        protected Medium medium;
        protected String name;
        protected Types identify;
        JTextField SentText;
        JTextArea ReceiveArea;
    
        public Colleague(String name, Types identify) {
            //初始化弹窗标题
            super(name);
            //初始化同事类信息
            this.name = name;
            this.identify = identify;
            //初始化弹窗
            ClientWindow();
        }
    
        //定义客户端
        void ClientWindow() {
            //初始化容器
            Container cp = this.getContentPane();
            ReceiveArea = new JTextArea(10, 18);
            ReceiveArea.setEditable(false);
            SentText = new JTextField(18);
    
            // 接收消息模块
            JPanel p1 = new JPanel();
            p1.setBorder(BorderFactory.createTitledBorder("接收内容:"));
            p1.add(ReceiveArea);
            // 初始化消息内容
            JScrollPane sp = new JScrollPane(p1);
            // 置顶
            cp.add(sp, BorderLayout.NORTH);
    
            // 发送消息模块
            JPanel p2 = new JPanel();
            p2.setBorder(BorderFactory.createTitledBorder("发送内容:"));
            p2.add(SentText);
            // 置底
            cp.add(p2, BorderLayout.SOUTH);
            // 输入框设置监听
            SentText.addActionListener(this);
    
            // 设置客户端窗口属性
            this.setLocation(50, 100); // 窗口位置
            this.setSize(250, 330);// 窗口大小
            this.setResizable(false); // 窗口大小不可调整
            this.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); // 窗口关闭操作-退出
            this.setVisible(true); // 窗口可见
        }
    
        @Override
        public void actionPerformed(ActionEvent e) {
            // 发送消息
            String tempInfo = SentText.getText().trim();
            this.send(tempInfo);
            // 输入框重置为空
            SentText.setText("");
        }
    
        @Override
        public String getName() {
            return name;
        }
    
        public void setMedium(Medium medium) {
            this.medium = medium;
        }
    
        public abstract void send(String ad);
    
        public abstract void receive(Colleague from, String ad);
    }
    
    //具体同事类
    class CommonColleague extends Colleague {
        private static final long serialVersionUID = -1443076716629516027L;
    
        public CommonColleague(String name, Types identify) {
            super(name, identify);
        }
    
        public void send(String msg) {
            // 显示消息
            ReceiveArea.append("我: " + msg + "
    ");
            // 使滚动条滚动到最底端
            ReceiveArea.setCaretPosition(ReceiveArea.getText().length());
            // 转发消息
            medium.relay(this, msg);
        }
    
        public void receive(Colleague from, String msg) {
            // 获取发送者身份
            String sender = from.name + "(" + from.identify.getTypeName() + ")";
            // 显示消息
            ReceiveArea.append(sender + ": " + msg + "
    ");
            // 使滚动条滚动到最底端
            ReceiveArea.setCaretPosition(ReceiveArea.getText().length());
        }
    }
    
    //具体同事类:学生
    class Student extends CommonColleague {
        private String stuNum;// 学号
        private String classId;// 班级编号
    
        //身份
        private final static Types identify = Types.Student;
    
        public Student(String name) {
            super(name, identify);
        }
    }
    
    //具体同事类:老师
    class Teacher extends CommonColleague {
        private String phone;// 手机号
    
        //身份
        private final static Types identify = Types.Teacher;
    
        public Teacher(String name) {
            super(name, identify);
        }
    }
    
    //具体同事类:管理员
    class Manager extends CommonColleague {
        //身份
        private final static Types identify = Types.Manager;
    
        public Manager(String name) {
            super(name, identify);
        }
    }
    

    运行结果:

    image-20210225115154936


    相关demo地址https://gitee.com/echo_ye/practice/tree/master/src/main/java/com/company/designPattern/mediator


    4.实际使用

    在实际开发中,通常并不严格按照上述模式开发,而通过简化来使快速开发,并缩小项目体积

    1. 不定义中介者接口,直接使用中介者(不太建议):效果显而易见,但即便只有一个中介者也不太建议,使项目更简单很重要,但易于扩展同样重要

    2. 同事类不持有中介者,使用时直接获取并调用(建议):相互持有是一个很有风险的事情,避开风险显然有利无害

    3. 单例化中介者(建议):若只有一个中介者,将其单例化可以避免误操作再次实例化;当然多个中介者的话就不能这样了


    后记

    中介者模式的核心目的是将网状结构转换为星型结构,即从N-N改变为N-1-N,从而降低系统的复杂度


    作者:Echo_Ye

    WX:Echo_YeZ

    Email :echo_yezi@qq.com

    个人站点:在搭了在搭了。。。(右键 - 新建文件夹)

  • 相关阅读:
    线程等待和通知
    什么是代码?code?
    代理(正向代理)和反向代理的区别是什么?
    什么是代理,什么是代理服务器,使用代理服务器的目的是什么?
    计算领域,编码的含义到底是什么?
    linux中,通过crontab -e编辑生成的定时任务,写在哪个文件中
    linux,shell中if else if的写法,if elif
    linux,shell脚本中获取脚本的名字,使用脚本的名字。
    linux,crontab定时任务中为脚本指定使用参数,crontab的脚本中是否可以带参数
    ssh在本地调用远程主机上的命令,不登录远程主机shell
  • 原文地址:https://www.cnblogs.com/silent-bug/p/14446611.html
Copyright © 2011-2022 走看看