【设计模式(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 + "\n");
        // 使滚动条滚动到最底端
        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 + "\n");
        // 使滚动条滚动到最底端
        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

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

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 201,681评论 5 474
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,710评论 2 377
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 148,623评论 0 334
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,202评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,232评论 5 363
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,368评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,795评论 3 393
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,461评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,647评论 1 295
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,476评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,525评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,226评论 3 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,785评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,857评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,090评论 1 258
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,647评论 2 348
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,215评论 2 341

推荐阅读更多精彩内容