JS如何实现中介者模式?中介者的作用

中介者模式通过引入中间人协调对象交互,降低组件间耦合度,提升可维护性与复用性,适用于复杂表单等多组件联动场景。

js如何实现中介者模式?中介者的作用

中介者模式在JavaScript里,简单来说,就是引入一个“中间人”角色,专门负责协调一组对象之间的交互。它把原本可能错综复杂的直接点对点通信,集中到一个地方处理,让各个对象之间不再直接打交道,从而大大降低了它们之间的耦合度。它的核心作用,就是让系统里的组件能更独立地发展,同时又能高效地协作。

要实现中介者模式,我们通常会定义一个中介者对象,以及若干个“同事”对象。这些同事对象不再直接引用或调用其他同事对象的方法,而是通过中介者来发送消息或请求服务。中介者接收到请求后,根据内部逻辑决定如何转发或处理,再通知相关的同事。

举个例子,想象一个复杂的表单,里面有多个输入框、下拉菜单和按钮,它们之间有很多联动逻辑:一个输入框的值变化可能影响另一个下拉菜单的可用选项,一个按钮的点击可能需要根据多个输入框的状态来决定是否启用。如果没有中介者,这些组件会直接互相监听和操作,代码很快就变成一团乱麻。

用中介者模式来组织,大概是这样:

// 中介者class FormMediator {    constructor() {        this.components = {}; // 存储所有注册的组件    }    register(name, component) {        this.components[name] = component;        component.setMediator(this); // 让组件知道自己的中介者是谁    }    notify(sender, event) {        // 根据发送者和事件类型,决定通知哪些组件        switch (event) {            case 'inputChanged':                if (sender.name === 'usernameInput') {                    // 假设用户名输入后,需要验证并可能影响提交按钮状态                    const username = this.components['usernameInput'].getValue();                    if (username.length > 5) {                        this.components['submitButton'].enable();                    } else {                        this.components['submitButton'].disable();                    }                }                break;            case 'dropdownSelected':                if (sender.name === 'countryDropdown') {                    const country = this.components['countryDropdown'].getValue();                    // 根据国家选择,更新城市下拉列表                    this.components['cityDropdown'].updateOptions(country);                }                break;            // 更多事件处理...        }    }}// 同事组件基类 (为了演示,简化处理)class Component {    constructor(name) {        this.name = name;        this.mediator = null;    }    setMediator(mediator) {        this.mediator = mediator;    }    // 抽象方法,子类实现具体业务逻辑    // 当自身状态变化时,通知中介者    changed(event) {        if (this.mediator) {            this.mediator.notify(this, event);        }    }}// 具体同事组件class InputField extends Component {    constructor(name, value = '') {        super(name);        this._value = value;    }    getValue() {        return this._value;    }    setValue(newValue) {        this._value = newValue;        console.log(`${this.name} 值变为: ${newValue}`);        this.changed('inputChanged'); // 状态变化时通知中介者    }}class Dropdown extends Component {    constructor(name, options = []) {        super(name);        this._selected = options[0] || null;        this.options = options;    }    getValue() {        return this._selected;    }    select(value) {        if (this.options.includes(value)) {            this._selected = value;            console.log(`${this.name} 选中: ${value}`);            this.changed('dropdownSelected');        }    }    updateOptions(basedOn) {        // 模拟根据其他组件状态更新选项        console.log(`${this.name} 根据 ${basedOn} 更新了选项。`);        // 实际中这里会有复杂的逻辑来获取新的选项        this.options = basedOn === 'USA' ? ['New York', 'Los Angeles'] : ['London', 'Paris'];        this._selected = this.options[0]; // 默认选中第一个    }}class Button extends Component {    constructor(name, text) {        super(name);        this.text = text;        this._enabled = false;    }    enable() {        this._enabled = true;        console.log(`${this.name} 按钮已启用。`);    }    disable() {        this._enabled = false;        console.log(`${this.name} 按钮已禁用。`);    }    click() {        if (this._enabled) {            console.log(`${this.name} 按钮被点击。`);            this.changed('buttonClicked'); // 通知中介者,可能触发提交        } else {            console.log(`${this.name} 按钮被点击,但当前禁用。`);        }    }}// 使用示例const mediator = new FormMediator();const usernameInput = new InputField('usernameInput');const countryDropdown = new Dropdown('countryDropdown', ['USA', 'UK', 'France']);const cityDropdown = new Dropdown('cityDropdown'); // 初始为空const submitButton = new Button('submitButton', '提交');mediator.register('usernameInput', usernameInput);mediator.register('countryDropdown', countryDropdown);mediator.register('cityDropdown', cityDropdown);mediator.register('submitButton', submitButton);// 模拟用户操作usernameInput.setValue('john_doe_123'); // 触发中介者,可能启用提交按钮countryDropdown.select('USA'); // 触发中介者,更新城市下拉列表submitButton.click(); // 尝试点击usernameInput.setValue('jo'); // 再次改变值,可能禁用提交按钮submitButton.click(); // 再次尝试点击

这段代码展示了核心思想:所有组件的交互都通过

FormMediator

来协调。组件自身只关心自己的状态变化并通知中介者,而无需知道其他组件的存在。

中介者模式解决了哪些痛点?

在我看来,中介者模式最显著的价值在于它能有效缓解“网状”依赖的困境。在一个大型或复杂的系统中,如果每个组件都需要直接知道并操作其他多个组件,那代码的耦合度会高到令人发指。这种情况下,修改一个组件可能会牵一发而动全身,维护成本呈指数级增长。

它解决的痛点主要体现在几个方面:

降低耦合度:这是最直接的好处。组件之间不再直接通信,而是通过中介者间接沟通。这使得每个组件可以独立地进行开发、测试和修改,而不会对其他组件产生意料之外的影响。就像一个交通指挥员,车辆只和指挥员打交道,不用关心其他车辆的具体位置和意图。提高可维护性:当所有交互逻辑都集中在中介者中时,调试和理解系统行为变得容易得多。你不需要在多个组件之间跳来跳去追踪调用链,只需要查看中介者的

notify

方法就能大致了解系统的协作流程。这对于团队协作和新成员理解项目尤其重要。简化对象交互:原本复杂的N对N交互,现在变成了N对1(所有同事都只与中介者交互)和1对N(中介者与所有同事交互)的模式。这种结构上的简化,让系统的整体架构更加清晰。促进组件复用:因为组件不再直接依赖于特定其他组件,它们变得更加通用。你可以在不同的上下文或项目中复用这些组件,只要为它们提供一个新的中介者实例即可。

我个人在处理一些前端复杂表单或仪表盘时,

以上就是JS如何实现中介者模式?中介者的作用的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/111961.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年11月23日 18:00:14
下一篇 2025年11月23日 18:33:50

相关推荐

  • 利用PHPMyAdmin执行SQL语句创建数据库视图

    登录phpmyadmin并选择目标数据库;2. 点击顶部“sql”标签进入执行界面;3. 编写create view语句,例如:create view view_name as select columns from table where condition;4. 输入具体视图定义,如包含单表筛选…

    2025年12月11日 好文分享
    000
  • PHP基础语法有哪些重点?核心语法要点详细解析

    学php最先要掌握基础语法,规范代码从变量、控制结构、函数到数组操作四方面入手。1.变量与数据类型:理解弱类型特性,注意手动判断和转换类型,提升可读性与逻辑清晰度;2.控制结构:掌握if、switch、循环等结构,合理使用break、continue,避免嵌套混乱;3.函数定义与调用:学会封装逻辑,…

    2025年12月11日 好文分享
    000
  • PHP中的设计模式:如何实现单例和工厂模式

    单例模式通过私有化构造函数、静态实例和获取实例的方法确保类只有一个实例;工厂模式封装对象创建逻辑,客户端无需指定具体类;traits可复用单例逻辑并防止克隆与反序列化;依赖注入容器是工厂模式的高级形式,自动管理依赖;选择设计模式需考虑问题复杂性、可维护性、灵活性及团队熟悉度。1.单例模式控制实例化过…

    2025年12月11日 好文分享
    000
  • PHPCMS逻辑漏洞的发现与分析方法

    phpcms逻辑漏洞的发现与分析需遵循一套系统性流程。①首先熟悉phpcms的业务逻辑与系统架构,包括用户注册、登录、内容发布、权限划分等核心功能,理解模块间的交互关系;②其次关注输入与输出边界,对get、post、http头、上传文件等内容进行校验与异常测试,识别潜在越权或数据篡改点;③采用逆向思…

    2025年12月11日 好文分享
    000
  • PHP游戏编程:基础图形渲染

    php可以用于游戏编程,但需结合前端技术实现图形渲染。1. php负责处理游戏逻辑、数据存储和用户交互;2. 图形渲染依赖html5 canvas或webgl;3. 用户输入通过表单或ajax发送至php处理并更新游戏状态;4. 性能优化包括减少网络传输、使用opcode缓存、高效算法及前端渲染优化…

    2025年12月11日 好文分享
    000
  • 配置PhpStorm代码折叠和展开的规则

    phpstorm 的代码折叠功能可通过设置和快捷键开启或关闭,并支持按语言结构自定义折叠规则,同时提供快捷键与鼠标操作实现高效代码浏览。具体包括:1. 在 settings 中勾选 enable code folding 或使用快捷键切换状态;2. 在 code folding 设置项中启用或禁用不…

    2025年12月11日 好文分享
    000
  • 图片水印如何添加?GD库图像处理详细教程

    使用php的gd库可以灵活地为图片添加水印。1. 首先确保环境支持gd库,并建议使用png格式的水印图片;2. 使用imagettftext()函数可直接添加文字水印,通过imagecolorallocate()设置颜色和透明度;3. 通过加载主图和水印图并调用imagecopymerge()函数实…

    2025年12月11日 好文分享
    000
  • 解决PHP文件上传中“权限拒绝”错误:文件系统权限配置指南

    本教程详细探讨了PHP move_uploaded_file函数在文件上传过程中常见的“权限拒绝”错误。该错误通常是由于目标目录缺乏Web服务器用户所需的写入权限所致。文章将指导您诊断问题、理解文件系统权限(chmod和chown)的重要性,并提供安全且有效的解决方案,确保您的PHP文件上传功能顺利…

    2025年12月11日
    000
  • 解决 PHP move_uploaded_file 权限拒绝错误:文件上传指南

    在使用 PHP move_uploaded_file 函数处理文件上传时,开发者常遇到“权限拒绝”错误,导致文件无法保存到服务器指定目录。这通常是由于运行 PHP 进程的Web服务器用户对目标目录没有写入权限所致。解决此问题需要正确配置目录的文件系统权限,确保Web服务器用户拥有读写目标目录的权限,…

    2025年12月11日
    000
  • PHP move_uploaded_file 失败:权限问题解析与解决方案

    本文详细解析了PHP中使用move_uploaded_file函数上传文件时,因Permission denied错误导致文件保存失败的常见问题。教程将深入探讨该错误的根本原因——目标目录的写入权限不足,并提供了针对Linux/Unix系统下文件权限配置的实用解决方案,确保PHP文件上传功能稳定运行…

    2025年12月11日
    000
  • PHP怎么实现文件自动归档 文件智能归档方案分享

    文件自动归档可通过php实现,核心步骤包括扫描目录、制定规则和执行操作。1. 扫描目录时,推荐使用recursivedirectoryiterator和recursiveiteratoriterator替代scandir(),以提升性能并支持递归遍历;2. 制定归档规则可采用配置文件(如json),…

    2025年12月11日 好文分享
    000
  • PHP怎么实现文件批量解压 文件批量解压的4个操作技巧

    要实现php批量解压文件,首先需扫描目录获取所有压缩文件,接着根据文件类型选择合适的解压方法(如ziparchive类或shell命令),并在解压过程中处理异常;其次为避免文件覆盖,可创建独立子目录或添加时间戳;对于大型文件,应增加内存限制、设置超时时间或使用系统命令行工具;安全性方面,需验证文件类…

    2025年12月11日 好文分享
    000
  • PHP命名空间:组织代码结构

    php命名空间用于解决类名、函数名等标识符冲突问题,并提升代码可读性与维护性。1.命名空间通过逻辑分组避免冲突,如同不同文件夹允许同名文件;2.使用namespace声明命名空间,如namespace myappmodels;3.引用类时可用fqn或use关键字导入简化;4.支持子命名空间嵌套,如m…

    2025年12月11日 好文分享
    000
  • 配置PhpStorm代码格式化的规则和快捷键

    配置phpstorm的代码格式化规则和快捷键需先选择语言规范并设置代码风格,再自定义细节规则,最后配置快捷键及自动保存选项。首先打开settings进入editor > code style选择对应语言并新建或复制配置方案,可导入.editorconfig或psr-12标准,也可手动调整缩进、…

    2025年12月11日 好文分享
    000
  • PHP怎样解析3D模型文件 PHP解析STL/OBJ格式的3种方案

    php解析3d模型文件(如stl和obj格式)的核心方法包括:1. 使用纯php逐行读取并解析文本内容,提取顶点、面等信息;2. 对于二进制stl使用unpack()函数解析;3. 利用php扩展提升性能,但开发难度较高;4. 调用外部程序如python脚本进行解析。优化方式包括一次性读取文件、使用…

    2025年12月11日 好文分享
    000
  • PHP怎么处理图片水印 PHP添加水印的完整实现教程

    php处理图片水印需选择gd库或imagemagick扩展,gd库简单但功能有限,imagemagick功能强但配置复杂;添加水印时应避开关键区域,通常选右下角或左下角,透明度建议20%-50%,使用imagecopymerge()函数控制透明度;处理不同图片类型需根据文件类型调用对应函数如imag…

    2025年12月11日 好文分享
    000
  • Dcat Admin表单多行布局下,Radio单选按钮如何实现联动?

    Dcat Admin表单多行布局下Radio单选按钮联动解决方案 在Dcat Admin框架下构建表单时,实现表单元素联动,例如根据单选按钮(Radio)的选择动态显示或隐藏其他字段,经常会遇到问题,尤其是在多行布局下。本文将解决Dcat Admin多行布局中Radio单选按钮联动失效的问题。 问题…

    2025年12月11日
    000
  • 使用phpMyAdmin快速创建和管理数据库表

    phpmyadmin是一个基于web的mysql数据库管理工具,它提供图形界面,简化数据库操作。使用它创建数据库表的方法是:1. 选择数据库;2. 点击“新建”按钮;3. 定义表名、字段名、数据类型和长度等;4. 点击“保存”。phpmyadmin将操作转换成sql语句执行,同时支持数据导入导出和表…

    2025年12月11日
    000
  • Go语言如何实现动态方法调用?

    Go语言动态方法调用:巧用反射机制实现灵活调用 Go语言的静态特性决定了其方法名在编译期即已确定,因此无法直接使用变量作为方法名进行调用,这与PHP等动态语言有所不同。然而,通过Go语言强大的反射机制,我们可以实现类似的动态方法调用效果。 本文将探讨如何利用反射机制在Go中动态调用方法,并分析其优缺…

    2025年12月11日
    000
  • 白天夜晚模式切换后刷新页面就失效了,如何解决?

    白天/夜晚模式切换失效的解决方法 本文将解决白天/夜晚模式切换后刷新页面失效的问题。 问题在于,模式设置没有持久化保存,刷新页面后丢失了之前的选择。以下提供一种改进方案,利用cookie存储模式设置,并在页面加载时读取cookie恢复模式。 问题描述: 一个白天/夜晚模式切换按钮,刷新页面后,选择的…

    2025年12月11日
    000

发表回复

登录后才能评论
关注微信