【VUE】从源码角度说清楚MVVM!实现v-model!真的很简单!🔥

简介: 前言大家好,我是HoMeTown,今天聊一聊老生长谈的Vue之双向数据绑定。

前言

大家好,我是HoMeTown,今天聊一聊老生长谈的Vue之双向数据绑定

What?

首先我们先看什么是双向数据绑定

对于不是很了解设计模式的朋友,你可以先理解一下单向数据绑定,就是把数据绑定到视图,每次触发操作修改了数据视图就会更新,数据 -> 视图,可以理解为MV数据驱动视图

举个🌰:

网络异常,图片无法展示
|

通过点击按钮set name,触发点击事件,手动更新变量name的值为HoMeTown,但是当我改变input输入框里的值,变量 name的值却不变,如下图:

网络异常,图片无法展示
|

那么双向数据绑定就是在单向的基础上,通过操作更新视图数据自动更新,那上面的🌰来讲,就是我输入Input,变量name的值动态改变。视图 -> 数据,可以理解为VM视图驱动数据

How?

Vue中的双向数据绑定由三个重要部分组成:

  • 数据层(Model):应用的数据及业务逻辑
  • 视图层(View):应用的展示效果,各类UI组件
  • 业务逻辑层(ViewModel):框架封装的核心,它负责将数据与视图关联起来

这个分层的架构方案,用专业术语来讲就是MVVM

ViewModel

ViewModel干了两件事儿:

  • 数据变化后更新视图
  • 视图变化后更新数据

它由两个重要部分组成:

  • 监听器(Ovserver):对所有数据的属性进行监听
  • 解析器(Compiler):对每个元素节点的指令进行扫描解析,根据指令模板替换数据,以及绑定相应的更新函数

动手实现

要做什么

Vue中,双向数据绑定的流程为:

  • new Vue()执行初始化,对data执行响应化处理,这个过程发生在Observe
  • 对模板执行编译,找到其中动态绑定的数据,从data中获取并初始化视图,这个过程发生在Compile
  • 定义一个更新函数和Watcher,将来对应数据变化时,Watcher调用更新函数
  • 由于data的某个属性在视图中可能出现N次,所以每个属性都需要一个Dep来管理多个Watcher
  • data中的数据一旦发生变化,首先会找到对应的Dep,然后通知这个Dep下所有的Watcher执行更新函数

参考下图:

网络异常,图片无法展示
|

Do it

首先定义一个Vue类, 做三件事

  • 数据劫持
  • 属性代理
  • 模板编译
class Vue {
    constructor(options) {
        this.$data = options.data
        this.$options = options
        // 数据劫持
        observe(this.$data)
        // 属性代理
        proxy(this.$data)
        // 模板编译
        compile(el, this)
    }
}
复制代码

接下来开始实现observe函数,做三件事

  • 递归data,劫持每一个
  • getter的时候收集
  • setter的时候通知执行
function observe(obj) {
    // 递归终止条件
    if(!obj || typeof obj !== 'object') return // 是空的 && 不是一个对象
    Object.keys(obj).forEach( key => {
        // 当前key对应的value
        const value = obj[key]
        // value能到这里,有可能是object,需要递归劫持
        observe(value)
        // 为当前的key所对应的属性添加getter & setter
        Object.defineProrerty(obj, key, {
            // 当且仅当该属性的 `enumerable` 键值为 `true` 时,该属性才会出现在对象的枚举属性中。
            enumerable: true,
            // 当且仅当该属性的 `configurable` 键值为 `true` 时,该属性的描述符才能够被改变,同时该属性也能从对应的对象上被删除。
            configurable: true,
            get() {
                // 将new出来的Watcher实例进行收集
                Dep.target ? dep.addSub(Dep.target) : null
            },
            set(newValue) {
                if( val !== newValue ) dep.notify() // 通知执行
            }
        })
    })
}
复制代码

接下来实现Dep类,做两件事儿:

  • 依赖收集
  • 通知执行
//依赖收集的类
class Dep {
    constructor() {
        // 所有Watcher实例存在这里
        this.subs = []
    }
    // 添加Watcher实例
    addSub(watcher) {
        this.subs.push(watcher)
    }
    // 通知Watcher实例执行更新函数
    notify() {
        this.subs.forEach( w => w.update())
    }
}
复制代码

接下来实现订阅者Watcher类,做两件事:

  • 提供Dep.target
  • 提供更新数据的方法
class Watcher {
    // callback中,记录了当前watcher如何更新自己的文本内容
    // 与此同时,需要拿到最新的数据,所以,在new Watcher的时候,需要传递vm进来
    // 因为需要知道在vm很多属性中,哪个数据,才是当前自己所需要的数据,所以,new Watcher的时候,需要指定key
    constructor(vm, key, callback) {
        this.vm = vm
        this.key = key
        this.callback = callback
        // 把创建的watcher实例,在Dep.addSub时,存进Dep的subs里
        Dep.target = this; // 自定义target属性
        key.split(".").reduce((newobj, k) => newobj[k], vm);
        Dep.target = null;
    }
    // 发布者通知Watcher更新的方法
    update() {
        const value = this.key.split(".").reduce((newobj, k) => newobj[key], this.vm);
        this.callback(value)
    }
}
复制代码

最后实现compile,对HTML结构进行模板编译的方法:

function compile(el, vm) {
    // 获取elDom元素
    vm.$el = document.querySelector(el);
    // 创建文档碎片,提高Dom操作性能
    const fragment = document.createDocumentFragment();
    // 取出来
    while ((childNode = vm.$el.firstChild)) {
      fragment.appendChild(childNode);
    }
    // 进行模板编译
    replace(fragment)
    // 放进去呀
    vm.$el.appendChild(fragement)
    function replace(node) {
      // 定义匹配插值表达式的正则
      const regMustache = /\{\{\s*(\S+)\s*\}\}/;
      // \S匹配任何非空白字符
      // \s匹配任何空白字符,包括空格、制表符、换页符等等。
      // ()非空白字符提取出来,用一个小括号进行分组
      // 当前的node节点是一个文本子节点,需要进行替换
      if(node.nodeType == 3) {
          const text = node.textContent // 文本子节点的字符串内容
          const execResult = regMustache.exec(text) //为一个数组,索引为0的为{{name}},为1的为name,exec() 方法用于检索字符串中的正则表达式的匹配。
          if(execResult) {
              const value = execResult[1].split(".").reduce((newobj, k) => newobj[k], vm)
              node.textContent = text.replace(regMustache, value)
              // 此时,就可以创建Watcher实例,将这个方法存到watcher上,调用update就执行
              new Watcher(vm, execResult[1], (newValue) => {
                  node.textContent = text.replace(regMustache, newValue)
              });
              // good good
          }
          // 递归结束
          return
      } 
      // 判断当前的node节点是否为input输入框
      if(node.nodeType === 1 && node.tagName.toUpperCase() === 'INPUT' ){
          // 首先要做v-model,就得先拿到属性节点
          const attrs = Array.from(node.attributes);
          const findResult = attrs.find((x) => x.name === "v-model");
          if(findResult) {
              // 当前有v-model,获取值
              const expStr = findResult.value;
              const value = expStr.split(".").reduce((newobj, k) => newobj[k], vm);
              node.value = value;
              // 创建Watcher实例
              new Watcher(vm, expStr, (newValue) => {
                  node.value = newValue
              })
              // 监听input事件,拿到文本框最新的值,然后更新到vm上
              node.addEventListener("input", e => {
                  const keys = expStr.split(".")
                  const keysLen = keys.length
                  const obj = keys.slice(0, keysLen - 1).reduce((newobj, k) => newobj[k], vm);
                  obj[keys[keysLen - 1]] = e.target.value
              })
          }
      }
      // 走到这,证明不是文本节点,递归处理
      node.childNodes.forEach( child => replace(child))
    }
}
复制代码

测试

还是用最开始我们的那个🌰,修改如下:

HTML

<div id="app">
  <p>name:<span id="nameBox">{{name}}</span></p>
  <input v-model="name" id="ipt" type="text" />
  <button id="set">Set name</button>
</div>
复制代码

JS

const vm = new Vue({
    el: "#app",
    data: {
      name: "No name yet!",
    },
  });
  const setBtn = document.getElementById("set");
  setBtn.onclick = function () {
    vm.name = "Is HoMeTown!!";
  };
复制代码

点击按钮,修改Vue实例vm的属性name = 'Is HoMeTown!!'

网络异常,图片无法展示
|

可以看到已经成功了!这是单向,然后我们试一试,修改输入框的内容,上方name的值不会不跟着改变:

网络异常,图片无法展示
|

SUCCESS!!!!!!!

总结

Vue中,双向数据绑定的原理总结的来说有几点:

  • observe 进行数据劫持,getter时添加Watcher,setter时通知Watcher.update
  • Dep类实现 依赖收集与通知执行
  • Watcher类实现 订阅者执行更新
  • compile 进行模板编译,解析v-model,给input添加事件

完结~


目录
相关文章
|
6天前
|
JavaScript
vue使用iconfont图标
vue使用iconfont图标
51 1
|
2月前
|
缓存 JavaScript UED
Vue3中v-model在处理自定义组件双向数据绑定时有哪些注意事项?
在使用`v-model`处理自定义组件双向数据绑定时,要仔细考虑各种因素,确保数据的准确传递和更新,同时提供良好的用户体验和代码可维护性。通过合理的设计和注意事项的遵循,能够更好地发挥`v-model`的优势,实现高效的双向数据绑定效果。
147 64
|
2月前
|
JavaScript 前端开发 API
Vue 3 中 v-model 与 Vue 2 中 v-model 的区别是什么?
总的来说,Vue 3 中的 `v-model` 在灵活性、与组合式 API 的结合、对自定义组件的支持等方面都有了明显的提升和改进,使其更适应现代前端开发的需求和趋势。但需要注意的是,在迁移过程中可能需要对一些代码进行调整和适配。
117 60
|
17天前
|
JavaScript 关系型数据库 MySQL
基于VUE的校园二手交易平台系统设计与实现毕业设计论文模板
基于Vue的校园二手交易平台是一款专为校园用户设计的在线交易系统,提供简洁高效、安全可靠的二手商品买卖环境。平台利用Vue框架的响应式数据绑定和组件化特性,实现用户友好的界面,方便商品浏览、发布与管理。该系统采用Node.js、MySQL及B/S架构,确保稳定性和多功能模块设计,涵盖管理员和用户功能模块,促进物品循环使用,降低开销,提升环保意识,助力绿色校园文化建设。
|
2月前
|
前端开发 JavaScript 测试技术
Vue3中v-model在处理自定义组件双向数据绑定时,如何避免循环引用?
Web 组件化是一种有效的开发方法,可以提高项目的质量、效率和可维护性。在实际项目中,要结合项目的具体情况,合理应用 Web 组件化的理念和技术,实现项目的成功实施和交付。通过不断地探索和实践,将 Web 组件化的优势充分发挥出来,为前端开发领域的发展做出贡献。
41 8
|
2月前
|
存储 JavaScript 数据管理
除了provide/inject,Vue3中还有哪些方式可以避免v-model的循环引用?
需要注意的是,在实际开发中,应根据具体的项目需求和组件结构来选择合适的方式来避免`v-model`的循环引用。同时,要综合考虑代码的可读性、可维护性和性能等因素,以确保系统的稳定和高效运行。
34 1
|
2月前
|
JavaScript
Vue3中使用provide/inject来避免v-model的循环引用
`provide`和`inject`是 Vue 3 中非常有用的特性,在处理一些复杂的组件间通信问题时,可以提供一种灵活的解决方案。通过合理使用它们,可以帮助我们更好地避免`v-model`的循环引用问题,提高代码的质量和可维护性。
44 1
|
2月前
|
JavaScript
在 Vue 3 中,如何使用 v-model 来处理自定义组件的双向数据绑定?
需要注意的是,在实际开发中,根据具体的业务需求和组件设计,可能需要对上述步骤进行适当的调整和优化,以确保双向数据绑定的正确性和稳定性。同时,深入理解 Vue 3 的响应式机制和组件通信原理,将有助于更好地运用 `v-model` 实现自定义组件的双向数据绑定。
|
2月前
|
JavaScript API 开发者
Vue是如何进行组件化的
Vue是如何进行组件化的
|
JavaScript 容器
【Vue源码解析】mustache模板引擎
【Vue源码解析】mustache模板引擎
73 0

热门文章

最新文章