观察者模式:定义了对象间一种一对多的依赖关系,当目标对象 Subject 的状态发生改变时,全部依赖它的对象 Observer 都会获得通知。
简单点:女神有男友了,朋友圈晒个图,甜蜜宣言 “老娘成功脱单,但愿大家欢喜”。各位潜藏备胎纷纷失恋,只能安慰本身你不是惟一一个。前端
Subject
,拥有方法:添加 / 删除 / 通知 Observer
;Observer
,拥有方法:接收 Subject
状态变动通知并处理;Subject
状态变动时,通知全部 Observer
。Subject
添加一系列 Observer
, Subject
负责维护与这些 Observer
之间的联系,“你对我有兴趣,我更新就会通知你”。vue
// 目标者类 class Subject { constructor() { this.observers = []; // 观察者列表 } // 添加 add(observer) { this.observers.push(observer); } // 删除 remove(observer) { let idx = this.observers.findIndex(item => item === observer); idx > -1 && this.observers.splice(idx, 1); } // 通知 notify() { for (let observer of this.observers) { observer.update(); } } } // 观察者类 class Observer { constructor(name) { this.name = name; } // 目标对象更新时触发的回调 update() { console.log(`目标者通知我更新了,我是:${this.name}`); } } // 实例化目标者 let subject = new Subject(); // 实例化两个观察者 let obs1 = new Observer('前端开发者'); let obs2 = new Observer('后端开发者'); // 向目标者添加观察者 subject.add(obs1); subject.add(obs2); // 目标者通知更新 subject.notify(); // 输出: // 目标者通知我更新了,我是前端开发者 // 目标者通知我更新了,我是后端开发者
观察者模式虽然实现了对象间依赖关系的低耦合,但却不能对事件通知进行细分管控,如 “筛选通知”,“指定主题事件通知” 。git
好比上面的例子,仅通知 “前端开发者” ?观察者对象如何只接收本身须要的更新通知?上例中,两个观察者接收目标者状态变动通知后,都执行了 update()
,并没有区分。github
“00后都在追求个性的时代,我能不能有点不同?”,这就引出咱们的下一个模式。进阶版的观察者模式。“发布订阅模式”,部分文章对二者是否同样都存在争议。编程
仅表明我的观点:两种模式很相似,可是仍是略有不一样,就是多了个第三者,因 JavaScript 非正规面向对象语言,且函数回调编程的特色,使得 “发布订阅模式” 在 JavaScript 中代码实现可等同为 “观察模式”。后端
发布订阅模式:基于一个事件(主题)通道,但愿接收通知的对象 Subscriber 经过自定义事件订阅主题,被激活事件的对象 Publisher 经过发布主题事件的方式通知各个订阅该主题的 Subscriber 对象。
发布订阅模式与观察者模式的不一样,“第三者” (事件中心)出现。目标对象并不直接通知观察者,而是经过事件中心来派发通知。设计模式
// 事件中心 let pubSub = { list: {}, subscribe: function (key, fn) { // 订阅 if (!this.list[key]) { this.list[key] = []; } this.list[key].push(fn); }, publish: function(key, ...arg) { // 发布 for(let fn of this.list[key]) { fn.call(this, ...arg); } }, unSubscribe: function (key, fn) { // 取消订阅 let fnList = this.list[key]; if (!fnList) return false; if (!fn) { // 不传入指定取消的订阅方法,则清空全部key下的订阅 fnList && (fnList.length = 0); } else { fnList.forEach((item, index) => { if (item === fn) { fnList.splice(index, 1); } }) } } } // 订阅 pubSub.subscribe('onwork', time => { console.log(`上班了:${time}`); }) pubSub.subscribe('offwork', time => { console.log(`下班了:${time}`); }) pubSub.subscribe('launch', time => { console.log(`吃饭了:${time}`); }) // 发布 pubSub.publish('offwork', '18:00:00'); pubSub.publish('launch', '12:00:00'); // 取消订阅 pubSub.unSubscribe('onwork');
发布订阅模式中,订阅者各自实现不一样的逻辑,且只接收本身对应的事件通知。实现你想要的 “不同”。api
DOM 事件监听也是 “发布订阅模式” 的应用:函数
let loginBtn = document.getElementById('#loginBtn'); // 监听回调函数(指定事件) function notifyClick() { console.log('我被点击了'); } // 添加事件监听 loginBtn.addEventListener('click', notifyClick); // 触发点击, 事件中心派发指定事件 loginBtn.click(); // 取消事件监听 loginBtn.removeEventListener('click', notifyClick);
发布订阅的通知顺序:post
on
和 trigger
,$.callback()
;$on/$emit
jQuery 的 $.Callback() 更像是观察者模式的应用,不能更细粒度管控。
function notifyHim(value) { console.log('He say ' + value); } function notifyHer(value) { console.log('She say ' + value); } $cb = $.Callbacks(); // 声明一个回调容器:订阅列表 $cb.add(notifyHim); // 向回调列表添加回调:订阅 $cb.add(notifyHer); // 向回调列表添加回调:订阅 $cb.fire('help'); // 调用全部回调: 发布
利用 Object.defineProperty()
对数据进行劫持,设置一个监听器 Observer
,用来监听数据对象的属性,若是属性上发生变化了,交由 Dep
通知订阅者 Watcher
去更新数据,最后指令解析器 Compile
解析对应的指令,进而会执行对应的更新函数,从而更新视图,实现了双向绑定。
Observer
(数据劫持)Dep
(发布订阅)Watcher
(数据监听)Compile
(模版编译)关于 Vue 双向数据绑定原理,可自行参考其它文章,或推荐本篇 《 vue双向数据绑定原理》。
Vue
中,父组件经过 props
向子组件传递数据(自上而下的单向数据流)。父子组件之间的通讯,经过自定义事件即 $on
, $emit
来实现(子组件 $emit
,父组件 $on
)。
原理其实就是 $emit
发布更新通知,而 $on
订阅接收通知。Vue
中还实现了 $once
(一次监听),$off
(取消订阅)。
// 订阅 vm.$on('test', function (msg) { console.log(msg) }) // 发布 vm.$emit('test', 'hi')
都是定义一个一对多的依赖关系,有关状态发生变动时执行相应的通知。
发布订阅模式更灵活,是进阶版的观察者模式,指定对应分发。
Object.defineProperty()
属性描述符;Proxy
代理;参考文章:
本文首发Github,期待Star!
https://github.com/ZengLingYong/blog
做者:以乐之名 本文原创,有不当的地方欢迎指出。转载请指明出处。