JavaScript设计模式(二十二):媒婆-中介者模式

简介: 媒婆-中介者模式

中介者模式(Mediator)

通过中介者对象封装一系列对象之间的交互,使对象之间不再相互引用,降低他们之间的耦合。有时中介者对象也可改变对象之间的交互。

需求:为导航模块添加设置层用来控制显示隐藏

image.png

<style>
    .icon {
    
    
        width: 30px;
        height: 30px;
        fill: currentColor;
        overflow: hidden;
    }

    .nav {
    
    
        width: 300px;
        height: 40px;
        padding-top: 10px;
        background-color: #fafafa;
        display: flex;
        justify-content: space-between;
        align-items: center;
        border: 1px solid #aaa;
    }

    .nav .nav-item {
    
    
        flex: 1;
        text-align: center;
    }

    .nav .nav-item span {
    
    
        cursor: pointer;
        position: relative;
        display: inline-block;
    }

    .nav .nav-item span b {
    
    
        position: absolute;
        top: -5px;
        right: -10px;
        font-size: 8px;
        padding: 1px 3px;
        background-color: red;
        color: #fff;
        border-radius: 10px;
    }
</style>

<body>
    <label for="hide_nav"><input id="hide_nav" checked type="checkbox" />显示/隐藏导航</label>
    <label for="hide_notice"><input id="hide_notice" checked type="checkbox" />显示/隐藏通知</label>
    <nav class="nav">
        <div id="home_nav" class="nav-item">
            <span>
                <svg class="icon" aria-hidden="true">
                    <use xlink:href="#icon-fangzi"></use>
                </svg>
                <b id="home_notice">10</b>
            </span>
        </div>
        <div id="community_nav" class="nav-item">
            <span>
                <svg class="icon" aria-hidden="true">
                    <use xlink:href="#icon-shequ"></use>
                </svg>
                <b id="community_notice">99+</b>
            </span>
        </div>
        <div id="me_nav" class="nav-item">
            <span>
                <svg class="icon" aria-hidden="true">
                    <use xlink:href="#icon-wode"></use>
                </svg>
                <b id="me_notice">24</b>
            </span>
        </div>
    </nav>
    <script src="./icons/iconfont.js"></script>
    <script src="./index.js"></script>
</body>
// 中介者对象
var Mediator = (function () {
   
   
    // 消息对象
    var _messages = {
   
   };
    return {
   
   
        _messages,
        /**
         * 订阅消息方法
         * @param {string} type 消息名称
         * @param {Function} action 消息回调函数
         */
        register(type, action) {
   
   
            // 如果该消息不存在,则建立该消息容器
            if (!_messages[type]) {
   
   
                _messages[type] = [];
            }
            // 存入新消息回调函数
            _messages[type].push(action);
        },
        /**
         * 发布消息方法
         * @param {*} type 消息名称
         */
        send(type) {
   
   
            // 如果该消息已经被订阅
            if (_messages[type]) {
   
   
                // 遍历已存储的消息回调函数
                _messages[type].forEach(action => action && action());
            }
        }
    };
}());

/**
 * 显示隐藏组件
 * @param {string[]} ids DOM元素
 * @param {boolean} isShow 显示隐藏
 */
const isShowNavWidget = function (ids, isShow) {
   
   
    ids.forEach(id => {
   
   
        const widget = document.getElementById(id);
        widget.style.visibility = isShow ? 'visible' : 'hidden';
    })
};

// 注册组件显示隐藏
(function () {
   
   
    // 显示导航 --- 不需要显示通知
    Mediator.register('showNav', function () {
   
   
        isShowNavWidget(['home_nav', 'community_nav', 'me_nav'], true);
    });

    // 隐藏导航 --- 必须隐藏通知
    Mediator.register('hideNav', function () {
   
   
        isShowNavWidget(['home_nav', 'community_nav', 'me_nav'], false);
    });
    Mediator.register('hideNav', function () {
   
   
        isShowNavWidget(['home_notice', 'community_notice', 'me_notice'], false);
    });

    // 显示通知 --- 必须显示导航
    Mediator.register('showNotice', function () {
   
   
        isShowNavWidget(['home_notice', 'community_notice', 'me_notice'], true);
    });
    Mediator.register('showNotice', function () {
   
   
        isShowNavWidget(['home_nav', 'community_nav', 'me_nav'], true);
    });

    // 隐藏通知 --- 不需要隐藏导航
    Mediator.register('hideNotice', function () {
   
   
        isShowNavWidget(['home_notice', 'community_notice', 'me_notice'], false);
    });
}());

// 设置层模块
(function () {
   
   
    const hideNav = document.getElementById('hide_nav');
    const hideNotice = document.getElementById('hide_notice');

    // 显示隐藏所有导航
    hideNav.onchange = function (e) {
   
   
        // 显示导航 --- 不需要显示通知
        if (hideNav.checked) {
   
   
            Mediator.send('showNav');
        }
        // 隐藏导航 --- 必须隐藏通知
        else {
   
   
            hideNotice.checked = false;
            Mediator.send('hideNav');
        }
    };
    // 显示隐藏所有通知
    hideNotice.onchange = function (e) {
   
   
        // 显示通知 --- 必须显示导航
        if (hideNotice.checked) {
   
   
            hideNav.checked = true;
            Mediator.send('showNotice');
        }
        // 隐藏通知 --- 不需要隐藏导航
        else {
   
   
            Mediator.send('hideNotice');
        }
    };
})();

console.log(Mediator);

特点:

  • 观察者模式相比
    • 观察者模式一样,中介者模式的主要业务也是通过模块间或者对象间的复杂通信,来解决模块间或对象间的耦合。
    • 中介者对象的本质是分装多个对象的交互,并且这些对象的交互一般都是在中介者内部实现的。
    • 观察者模式中,订阅者是双向的,既可以是消息的发布者,也可以是消息的订阅者。
    • 中介者模式中,订阅者是单向的,只能是消息的订阅者。而消息统一由中介者对象发布,所有的订阅者对象间接地被中介者管理。
  • 外观模式封装特性相比
    • 外观模式封装的目的是为了提供更简单的易用接口,而不会添加其他功能。
    • 中介者模式对多个对象交互地封装,且这些对象一般处于同一层面上,并且封装的交互在中介者内部。
目录
相关文章
|
1月前
|
设计模式 JavaScript 前端开发
JavaScript设计模式--访问者模式
【10月更文挑战第1天】
30 3
|
2月前
|
设计模式 Java
Java设计模式-中介者模式(20)
Java设计模式-中介者模式(20)
|
3月前
|
设计模式 JavaScript 前端开发
从工厂到单例再到策略:Vue.js高效应用JavaScript设计模式
【8月更文挑战第30天】在现代Web开发中,结合使用JavaScript设计模式与框架如Vue.js能显著提升代码质量和项目的可维护性。本文探讨了常见JavaScript设计模式及其在Vue.js中的应用。通过具体示例介绍了工厂模式、单例模式和策略模式的应用场景及其实现方法。例如,工厂模式通过`NavFactory`根据用户角色动态创建不同的导航栏组件;单例模式则通过全局事件总线`eventBus`实现跨组件通信;策略模式用于处理不同的表单验证规则。这些设计模式的应用不仅提高了代码的复用性和灵活性,还增强了Vue应用的整体质量。
50 1
|
3月前
|
设计模式 前端开发 Java
【十三】设计模式~~~行为型模式~~~中介者模式(Java)
文章详细介绍了中介者模式(Mediator Pattern),这是一种对象行为型模式,用于封装一系列对象的交互,降低系统耦合度,并简化对象之间的交互关系。通过案例分析、结构图、时序图和代码示例,文章展示了中介者模式的组成部分、实现方式和应用场景,并讨论了其优点、缺点和适用情况。
【十三】设计模式~~~行为型模式~~~中介者模式(Java)
|
3月前
|
设计模式 JavaScript 前端开发
小白请看 JS大项目宝典:设计模式 教你如何追到心仪的女神
小白请看 JS大项目宝典:设计模式 教你如何追到心仪的女神
|
4月前
|
设计模式 JavaScript Go
js设计模式【详解】—— 状态模式
js设计模式【详解】—— 状态模式
79 7
|
4月前
|
设计模式 JavaScript
js设计模式【详解】—— 桥接模式
js设计模式【详解】—— 桥接模式
67 6
|
4月前
|
设计模式 JavaScript
js设计模式【详解】—— 原型模式
js设计模式【详解】—— 原型模式
49 6
|
4月前
|
设计模式 JavaScript 算法
js设计模式【详解】—— 模板方法模式
js设计模式【详解】—— 模板方法模式
47 6
|
4月前
|
设计模式 JavaScript 前端开发
JavaScript进阶 - JavaScript设计模式
【7月更文挑战第7天】在软件工程中,设计模式是解决常见问题的标准解决方案。JavaScript中的工厂模式用于对象创建,但过度使用可能导致抽象过度和缺乏灵活性。单例模式确保唯一实例,但应注意避免全局状态和过度使用。观察者模式实现了一对多依赖,需警惕性能影响和循环依赖。通过理解模式的优缺点,能提升代码质量。例如,工厂模式通过`createShape`函数动态创建对象;单例模式用闭包保证唯一实例;观察者模式让主题对象通知多个观察者。设计模式的恰当运用能增强代码可维护性。
81 0

热门文章

最新文章

  • 1
    C++一分钟之-设计模式:工厂模式与抽象工厂
    42
  • 2
    《手把手教你》系列基础篇(九十四)-java+ selenium自动化测试-框架设计基础-POM设计模式实现-下篇(详解教程)
    46
  • 3
    C++一分钟之-C++中的设计模式:单例模式
    53
  • 4
    《手把手教你》系列基础篇(九十三)-java+ selenium自动化测试-框架设计基础-POM设计模式实现-上篇(详解教程)
    37
  • 5
    《手把手教你》系列基础篇(九十二)-java+ selenium自动化测试-框架设计基础-POM设计模式简介(详解教程)
    61
  • 6
    Java面试题:结合设计模式与并发工具包实现高效缓存;多线程与内存管理优化实践;并发框架与设计模式在复杂系统中的应用
    56
  • 7
    Java面试题:设计模式在并发编程中的创新应用,Java内存管理与多线程工具类的综合应用,Java并发工具包与并发框架的创新应用
    40
  • 8
    Java面试题:如何使用设计模式优化多线程环境下的资源管理?Java内存模型与并发工具类的协同工作,描述ForkJoinPool的工作机制,并解释其在并行计算中的优势。如何根据任务特性调整线程池参数
    49
  • 9
    Java面试题:请列举三种常用的设计模式,并分别给出在Java中的应用场景?请分析Java内存管理中的主要问题,并提出相应的优化策略?请简述Java多线程编程中的常见问题,并给出解决方案
    105
  • 10
    Java面试题:设计模式如单例模式、工厂模式、观察者模式等在多线程环境下线程安全问题,Java内存模型定义了线程如何与内存交互,包括原子性、可见性、有序性,并发框架提供了更高层次的并发任务处理能力
    75