沉淀js-设计模式-02

本文深入探讨了发布-订阅模式,一种对象间一对多依赖关系的设计模式,详细讲解了其与观察者模式的区别,以及如何在JavaScript中实现解耦的时间和空间特性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

发布-订阅模式

  • 什么是"发布-订阅模式"

发布-订阅模式定义了对象之间的一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖它的对象都可以得到通知。

  • 优点
  1. 时间解耦
  2. 空间解耦

发布-订阅模式与观察者模式的区别

  • 发布-订阅模式和观察者模式概念相似,但在发布-订阅模式中,发布者和订阅者之间多了一层中间件:一个被抽象出来的信息调度 中心。
  • 观察者模式由具体目标调度,每个被订阅的目标里面都需要有对观察者的处理,这种方法会导致两个对象间的耦合,造成代码的冗余。
  • 发布订阅模式中统一由调度中心进行处理,订阅者和发布者互不干扰,消除了发布者和订阅者之间的依赖。实现了解耦,对象实体之间的订阅不存在依赖关系。比如发布者发布了很多消息,但是不想所有的订阅者都接收到,就可以在调度中心做一些处理,类似于权限控制之类的。还可以做一些节流操作。
class Event {
    constructor (o) {
        if (!!o) {
            if (typeof o !== 'object') {
                throw new Error('o is not a object');
            }
            const b = Object.keys(o).every(k => Array.isArray(o[k]));
            if (!b) {
                throw new Error('o[key] is not array');
            }
        }
        this.events = o || {};
    }
    addWatch(key, fn) {
        const { events } = this;
        if (!events[key]) {
            this.events[key] = [];
        }
        this.events[key].push(fn);
    }
    noticeAll() {
        const key = Array.prototype.shift.call(arguments);
        const { events } = this; 
        const fns = events[key];
        if (!fns || fns.length === 0) {
            return false;
        }
        for (let fn of fns) {
            fn.apply(null, arguments);
        }
    }
    remove(key, fn) {
        const fns = this.events[key];
        if (!fns || !fn) {
            return false;
        }
        // 反向遍历,避免修改数组的下标
        for (let i = fns.length - 1; i >= 0; i--) {
            let f = fns[i];
            if (f == fn) {
                fns.splice(i, 1);
            }
        }
    }
}
const c = new Event({
    event00: [(f) => {
        console.log(`Event is, ${f}, at event00`)
    }]
});
// 绑定自定义事件和回调函数

c.addWatch("event01", fn1 = (f) => {
  console.log(`Event is, ${f}, at event01`);
})

c.addWatch("event02", fn2 = (f) => {
  console.log(`Price is, ${f}, at event02`);
})

c.noticeAll("event01", 1000);
c.noticeAll("event02", 2000);

c.remove("event01", fn1);

复制代码

观察者模式

  • 观察者模式:观察者(Observer)直接订阅(Subscribe)主题(Subject),而当主题被激活的时候,会触发(Fire Event)观察者里的事件。
  • 发布-订阅模式:订阅者(Subscriber)把自己想订阅的事件注册(Subscribe)到调度中心(Topic),当发布者(Publisher)发布该事件(Publish topic)到调度中心,也就是该事件触发时,由调度中心统一调度(Fire Event)订阅者注册到调度中心的处理代码。

class TaskLine {
    constructor (opt) {
        this.name = opt && opt.name || '';
        this.info = opt && opt.info || '';
        this.eventList = [];
    }
    subscribe(target, fn) {
        const { name } = this;
        if (target.name === name) {
            throw new Error('Can not subscribe self');
        }
        if (target.eventList.indexOf(fn) == -1) {
            target.eventList.push(fn);
        }
    }
    publish(n) {
        this.eventList.forEach(list => list(n));
    }
}
const taskLine1 = new TaskLine({
    name: 'task1',
    info: 1
})
const taskLine2 = new TaskLine({
    name: 'task2',
    info: 2
})
const taskLine3 = new TaskLine({
    name: 'task3',
    info: 3
})
taskLine1.subscribe(taskLine1, (n) => {
    console.log(`TaskLine1 help taskLine1 go`);
}) // can not subscribe self;
taskLine1.subscribe(taskLine2, (n) => {
    console.log(`TaskLine1 help taskLine2 go`);
})
taskLine1.subscribe(taskLine3, (n) => {
    console.log(`TaskLine1 help taskLine3 go`);
})
taskLine2.subscribe(taskLine3, (n) => {
    if (n > 1000) {
        console.log(`TaskLine2 help taskLine3 go`);
    } else {
        console.log(`TaskLine2: I'm busy`);
    }
})
taskLine3.publish(1000);
复制代码

下一期 沉淀js-设计模式-03

老铁欢迎指导github

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值