目录
1.什么是发布订阅模式
2.实现简单的发布订阅
3.收集更新函数
4.触发更新函数
5.总结
一个响应式数据可能会有多个视图部分都需要依赖,也就是响应式数据变化之后,需要执行的更新函数可能不止一个,对于这种情况,有必要学习一下发布订阅模式。
1.什么是发布订阅模式
发布订阅模式定义了一种一对多的依赖关系,让多个订阅者对象同时监听某一个主题对象。这个主题对象在自身状态变化时,会通知所有订阅者对象,使它们能够自动更新自己的状态。(盗用一张图)
现实生活中有很多类似的例子,比如我最近在网上看上一双鞋子,联系到卖家后,这双鞋已经卖光了,于是问卖家什么时候有货,卖家告诉我,要等一个星期后才有货,可以收藏店铺,等有货的时候再通知,所以我收藏了此店铺,但与此同时,其他人等也喜欢这双鞋,也收藏了该店铺;等来货的时候就依次会通知他们。这就是简单的发布订阅。
- 指定发布者
- 给发布者添加一个缓存列表,用于存放回调函数以便通知订阅者
- 最后发布消息的时候,发布者遍历缓存列表,依次触发里面存放的订阅者回调函数
2.实现简单的发布订阅
先从dom绑定事件来说,一般一个事件绑定一个回调函数,但是也可以绑定多个回调函数。只不过方式不同,第二种就可以绑定多个。
- dom.οnclick=function(){}
- dom.addEventListener(‘click’,()=>{})
实现简单的发布订阅
// 增加dep对象 用来收集依赖和触发依赖const dep = { map: Object.create(null), // 收集 collect(dataProp, updateFn) { if (!this.map[dataProp]) { this.map[dataProp] = [] } this.map[dataProp].push(updateFn) }, // 触发 trigger(dataProp) { this.map[dataProp] && this.map[dataProp].forEach(updateFn => { updateFn() }) }}
3.收集更新函数
用于更新dom的更新函数集中起来
// 编译函数 function compile() { let app = document.getElementById('app') // 1.拿到app下所有的子元素 const nodes = app.childNodes // [text, input, text] //2.遍历所有的子元素 nodes.forEach(node => { // nodeType为1为元素节点 if (node.nodeType === 1) { const attrs = node.attributes // 遍历所有的attrubites找到 v-model Array.from(attrs).forEach(attr => { const dirName = attr.nodeName const dataProp = attr.nodeValue console.log(dirName, dataProp) if (dirName === 'v-text') { console.log(`更新了${dirName}指令,需要更新的属性为${dataProp}`) node.innerText = data[dataProp] // 收集更新函数 dep.collect(dataProp, () => { node.innerText = data[dataProp] }) } }) } }) }
4.触发更新函数
当属性变化时,通过属性找到对应的更新函数列表
function defineReactive(data, key, value) { Object.defineProperty(data, key, { get() { return value }, set(newValue) { // 更新视图 if (newValue === value) return value = newValue // 再次编译要放到新值已经变化之后只更新当前的key dep.trigger(key) } })}
5.总结
这是在vue框架之下的发布订阅,其原理中提到了几个专业名词
observe
对象指的是把数据处理成响应式的对象watcher
指的其实就是数据变化之后的更新函数 (vue中的watcher有两种,一种是用来更新视图的watcher,一种是通过watch配置项声明的watcher)dep
指的就是使用发布订阅实现的收集更新函数和触发更新函数的对象
发布订阅模式的本质是解决一对多的问题,在vue中实现数据变化之后的精准更新。
vue作者编程功力之深厚,运用之巧妙,令人叹服!!!!!!