Vue.js - 响应式原理基础

227 阅读3分钟

Vue.js 响应式原理

1)数据响应式

  • 数据模型仅仅是普通的 JavaScript 对象,而当我们修改数据时,视图会进行更新,避免了频繁的 DOM 操作,提高开发效率

2)双向绑定

  • 数据改变,视图改变;视图改变,数据也随之改变
  • 我们可以使用 v-model 在表单元素上创建双向数据绑定

3)数据驱动是 Vue 最独特的特性之一

  • 开发过程中仅需要关注数据本身,不需要关心数据是如何渲染到视图

4)数据响应式核心原理 ---- Vue2 ---- Object.defineProperty

// 模拟 Vue 中的 data 选项
    let data = {
      msg: 'hello',
      count: 10
    }

    // 模拟 Vue 的实例
    let vm = {}

    proxyData(data)

    function proxyData(data) {
      // 遍历 data 对象的所有属性
      Object.keys(data).forEach(key => {
        // 把 data 中的属性,转换成 vm 的 setter/setter
        Object.defineProperty(vm, key, {
          enumerable: true,
          configurable: true,
          get () {
            console.log('get: ', key, data[key])
            return data[key]
          },
          set (newValue) {
            console.log('set: ', key, newValue)
            if (newValue === data[key]) {
              return
            }
            data[key] = newValue
            // 数据更改,更新 DOM 的值
            document.querySelector('#app').textContent = data[key]
          }
        })
      })
    }

    // 测试
    vm.msg = 'Hello World'
    console.log(vm.msg)

5)数据响应式核心原理 ---- Vue3 ---- Proxy

  • 直接监听对象,而非属性
  • ES6 中新增,IE 不支持,性能由浏览器优化
// 模拟 Vue 中的 data 选项
    let data = {
      msg: 'hello',
      count: 0
    }

    // 模拟 Vue 实例
    let vm = new Proxy(data, {
      // 执行代理行为的函数
      // 当访问 vm 的成员会执行
      get (target, key) {
        console.log('get, key: ', key, target[key])
        return target[key]
      },
      // 当设置 vm 的成员会执行
      set (target, key, newValue) {
        console.log('set, key: ', key, newValue)
        if (target[key] === newValue) {
          return
        }
        target[key] = newValue
        document.querySelector('#app').textContent = target[key]
      }
    })

    // 测试
    vm.msg = 'Hello World'
    console.log(vm.msg)

6) 发布订阅模式

  • 发布/订阅模式

  • 订阅者

  • 发布者

  • 信号中心

我们假定,存在一个 “信号中心”,某个任务执行完成,就向信号中心**“发布”** 一个信号,其他任务可以向信号中心 “订阅” 这个信号,从而知道什么时候自己可以开始执行。这就叫做 发布/订阅模式

  • 自定义事件

    / Vue 自定义事件
        let vm = new Vue()
        // { 'click': [fn1, fn2], 'change': [fn] }
    
        // 注册事件(订阅消息)
        vm.$on('dataChange', () => {
          console.log('dataChange')
        })
    
        vm.$on('dataChange', () => {
          console.log('dataChange1')
        })
        // 触发事件(发布消息)
        vm.$emit('dataChange')
    
  • 兄弟组件通信过程

// eventBus.js
// 事件中心
let eventHub = new Vue()

// ComponentA.vue
// 发布者
addTodo: function () {
  // 发布消息(事件)
  eventHub.$emit('add-todo', { text: this.newTodoText })
  this.newTodoText = ''
}

// ComponentB.vue
// 订阅者
created: function () {
  // 订阅消息(事件)
  eventHub.$on('add-todo', this.addTodo)
}
模拟 Vue 自定义事件的实现
  • 模拟 Vue 自定义事件的实现

     // 事件触发器
        class EventEmitter {
          constructor () {
            // { 'click': [fn1, fn2], 'change': [fn] }
            this.subs = Object.create(null) // 创建对象
          }
    
          // 注册事件
          $on (eventType, handler) {
            this.subs[eventType] = this.subs[eventType] || []
            this.subs[eventType].push(handler)
          }
    
          // 触发事件
          $emit (eventType) {
            if (this.subs[eventType]) {
              this.subs[eventType].forEach(handler => {
                handler()
              })
            }
          }
        }
    
        // 测试
        let em = new EventEmitter()
        em.$on('click', () => {
          console.log('click1')
        })
        em.$on('click', () => {
          console.log('click2')
        })
    
        em.$emit('click')
    

7)观察者模式

  • 观察者(订阅者)-- Watcher
    • update(): 当事件发生时,具体要做的事情
  • 目标(发布者)-- Dep
    • subs 数组:存储所有的观察者
    • addSub(): 添加观察者
    • notify(): 当事件发生时,调用所有观察者的 update() 方法
  • 没有事件中心
  • 模拟 观察者模式 实现
// 发布者-目标
    class Dep {
      constructor () {
        // 记录所有的订阅者
        this.subs = []
      }
      // 添加订阅者
      addSub (sub) {
        if (sub && sub.update) {
          this.subs.push(sub)
        }
      }
      // 发布通知
      notify () {
        this.subs.forEach(sub => {
          sub.update()
        })
      }
    }
    // 订阅者-观察者
    class Watcher {
      update () {
        console.log('update')
      }
    }

    // 测试
    let dep = new Dep()
    let watcher = new Watcher()

    dep.addSub(watcher)

    dep.notify()

总结:

  • 观察者模式 是由具体目标调度,比如当事件触发,Dep 就会去调用观察者的方法,所以观察者模式的订阅者与发布者之间是存在依赖
  • 发布/订阅模式 由统一调度中心调用,因此发布者与订阅者不需要对方的存在