JavaScript 设计模式(八):组合模式

作者 : 开心源码 本文共2451个字,预计阅读时间需要7分钟 发布时间: 2022-05-13 共201人阅读

组合模式

组合模式:又叫 “部分整体” 模式,将对象组合成树形结构,以表示 “部分-整体” 的层次结构。通过对象的多态性体现,使得客户对单个对象和组合对象的使用具备一致性。

生活小栗子:文件目录,DOM 文档树

模式特点

  1. 表示 “部分-整体” 的层次结构,生成 “树叶型” 结构;
  2. 一致操作性,树叶对象对外接口保存一致(操作与数据结构一致);
  3. 自上而下的的请求流向,从树对象传递给叶对象;
  4. 调用顶层对象,会自行遍历其下的叶对象执行。

树叶型结构

代码实现

树对象和叶对象接口统一,树对象添加一个缓存数组,存储叶对象。执行树对象方法时,将请求传递给其下叶对象执行。

// 树对象 - 文件目录class CFolder {    constructor(name) {        this.name = name;        this.files = [];    }    add(file) {        this.files.push(file);    }    scan() {        for (let file of this.files) {            file.scan();        }    }}// 叶对象 - 文件class CFile {    constructor(name) {        this.name = name;    }    add(file) {        throw new Error('文件下面不能再增加文件');    }    scan() {        console.log(`开始扫描文件:${this.name}`);    }}let mediaFolder = new CFolder('娱乐');let movieFolder = new CFolder('电影');let musicFolder = new CFolder('音乐');let file1 = new CFile('钢铁侠.mp4');let file2 = new CFile('再谈记忆.mp3');movieFolder.add(file1);musicFolder.add(file2);mediaFolder.add(movieFolder);mediaFolder.add(musicFolder);mediaFolder.scan();/* 输出:开始扫描文件:钢铁侠.mp4开始扫描文件:再谈记忆.mp3*/

CFolderCFile 接口保持一致。执行 scan() 时,若发现是树对象,则继续遍历其下的叶对象,执行 scan()

JavaScript 不同于其它静态编程语言,实现组合模式的难点是保持树对象与叶对象之间接口保持统一,可借助 TypeScript 定制接口规范,实现类型束缚。

// 定义接口规范interface Compose {    name: string,    add(file: CFile): void,    scan(): void}// 树对象 - 文件目录class CFolder implements Compose {    fileList = [];    name: string;    constructor(name: string) {        this.name = name;    }    add(file: CFile) {        this.fileList.push(file);    }    scan() {        for (let file of this.fileList) {            file.scan();        }    }}// 叶对象 - 文件class CFile implements Compose {    name: string;    constructor(name: string) {        this.name = name;    }    add(file: CFile) {        throw new Error('文件下面不能再增加文件');    }    scan() {        console.log(`开始扫描:${this.name}`)    }}let mediaFolder = new CFolder('娱乐');let movieFolder = new CFolder('电影');let musicFolder = new CFolder('音乐');let file1 = new CFile('钢铁侠.mp4');let file2 = new CFile('再谈记忆.mp3');movieFolder.add(file1);musicFolder.add(file2);mediaFolder.add(movieFolder);mediaFolder.add(musicFolder);mediaFolder.scan();/* 输出:开始扫描文件:钢铁侠.mp4开始扫描文件:再谈记忆.mp3*/

透明性的安全问题

组合模式的透明性,指的是树叶对象接口保持统一,外部调用时无需区分。但是这会带来少量问题,如上述文件目录的例子,文件(叶对象)下不可再增加文件,因而需在文件类的 add() 方法中抛出异常,以作提示。

误区规避

1. 组合不是继承,树叶对象并不是父子对象

组合模式的树型结构是一种 HAS-A(聚合)的关系,而不是 IS-A 。树叶对象能够合作的关键,是它们对外保持统一接口,而不是叶对象继承树对象的属性方法,两者之间不是父子关系。

2. 叶对象操作保持一致性

叶对象除了与树对象接口一致外,操作也必需保持一致性。一片叶子只能生在一颗树上。调用顶层对象时,每个叶对象只能接收一次请求,一个叶对象不能从属多个树对象。

3. 叶对象实现冒泡传递

请求传递由树向叶传递,假如想逆转传递过程,需在叶对象中保留对树对象的引用,冒泡传递给树对象解决。

4. 不只是简单的子集遍历

调用对象的接口方法时,假如该对象是树对象,则会将请求传递给叶对象,由叶对象执行方法,以此类推。不同于迭代器模式,迭代器模式遍历并不会做请求传导。

应用场景

  1. 优化解决递归或者分级数据结构(文件系统 – 目录文件管理);
  2. 与其它设计模式联用,如与命令模式联用实现 “宏命令”。

优缺点

  • 优点:
    • 忽略组合对象和单个对象的差别,对外一致接口使用;
    • 解耦调用者与复杂元素之间的联络,解决方式变得简单。
  • 缺点
    • 树叶对象接口一致,无法区分,只有在运行时方可辨别;
    • 包裹对象创立太多,额外添加内存负担。

参考文章

  • 《JavaScript 设计模式与开发实践》

本文首发Github,期待Star!
ZengLingYong/blog

作者:以乐之名
本文原创,有不当的地方欢迎指出。转载请指明出处。

说明
1. 本站所有资源来源于用户上传和网络,如有侵权请邮件联系站长!
2. 分享目的仅供大家学习和交流,您必须在下载后24小时内删除!
3. 不得使用于非法商业用途,不得违反国家法律。否则后果自负!
4. 本站提供的源码、模板、插件等等其他资源,都不包含技术服务请大家谅解!
5. 如有链接无法下载、失效或广告,请联系管理员处理!
6. 本站资源售价只是摆设,本站源码仅提供给会员学习使用!
7. 如遇到加密压缩包,请使用360解压,如遇到无法解压的请联系管理员
开心源码网 » JavaScript 设计模式(八):组合模式

发表回复