目录

1.什么是发布订阅模式

2.实现简单的发布订阅

3.收集更新函数

4.触发更新函数

5.总结


一个响应式数据可能会有多个视图部分都需要依赖,也就是响应式数据变化之后,需要执行的更新函数可能不止一个,对于这种情况,有必要学习一下发布订阅模式。

1.什么是发布订阅模式

发布订阅模式定义了一种一对多的依赖关系,让多个订阅者对象同时监听某一个主题对象。这个主题对象在自身状态变化时,会通知所有订阅者对象,使它们能够自动更新自己的状态。(盗用一张图)

现实生活中有很多类似的例子,比如我最近在网上看上一双鞋子,联系到卖家后,这双鞋已经卖光了,于是问卖家什么时候有货,卖家告诉我,要等一个星期后才有货,可以收藏店铺,等有货的时候再通知,所以我收藏了此店铺,但与此同时,其他人等也喜欢这双鞋,也收藏了该店铺;等来货的时候就依次会通知他们。这就是简单的发布订阅。

  • 指定发布者
  • 给发布者添加一个缓存列表,用于存放回调函数以便通知订阅者
  • 最后发布消息的时候,发布者遍历缓存列表,依次触发里面存放的订阅者回调函数

2.实现简单的发布订阅

先从dom绑定事件来说,一般一个事件绑定一个回调函数,但是也可以绑定多个回调函数。只不过方式不同,第二种就可以绑定多个。

  1. dom.οnclick=function(){}
  2. 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框架之下的发布订阅,其原理中提到了几个专业名词

  1. observe对象指的是把数据处理成响应式的对象
  2. watcher指的其实就是数据变化之后的更新函数 (vue中的watcher有两种,一种是用来更新视图的watcher,一种是通过watch配置项声明的watcher)
  3. dep指的就是使用发布订阅实现的收集更新函数和触发更新函数的对象

发布订阅模式的本质是解决一对多的问题,在vue中实现数据变化之后的精准更新。

vue作者编程功力之深厚,运用之巧妙,令人叹服!!!!!!