Vue 双向数据绑定原理分析

作者: NARUTO_86 | 来源:发表于2016-07-24 21:17 被阅读32007次

关于双向数据绑定

当我们在前端开发中采用MV*的模式时,M - model,指的是模型,也就是数据,V - view,指的是视图,也就是页面展现的部分。通常,我们需要编写代码,将从服务器获取的数据进行“渲染”,展现到视图上。每当数据有变更时,我们会再次进行渲染,从而更新视图,使得视图与数据保持一致。也就是:

Paste_Image.png

而另一方面,页面也会通过用户的交互,产生状态、数据的变化,这个时候,我们则编写代码,将视图对数据的更新同步到数据,以致于同步到后台服务器。也就是:

Paste_Image.png

不同的前端 MV* 框架对于这种 Model 和 View 间的数据同步有不同的处理。在 Backbone 中,Model 到 View 的数据传递,可以在 View 中监听 Model 的 change 事件,每当 Model 更新,View 中重新执行 render。而 View 到 Model 的数据传递,可以监听 View 对应的 DOM 元素的各种事件,在检测到 View 状态变更后,将变更的数据发送到 Model。相较于 Backbone,AngularJS 所代表的 MVVM 框架则更进一步,从框架层面支持这种数据同步机制,而且是双向数据绑定:

Paste_Image.png

不过在不同的 MVVM 框架中,实现双向数据绑定的技术有所不同。

AngularJS 采用“脏值检测”的方式,数据发生变更后,对于所有的数据和视图的绑定关系进行一次检测,识别是否有数据发生了改变,有变化进行处理,可能进一步引发其他数据的改变,所以这个过程可能会循环几次,一直到不再有数据变化发生后,将变更的数据发送到视图,更新页面展现。如果是手动对 ViewModel 的数据进行变更,为确保变更同步到视图,需要手动触发一次“脏值检测”。

VueJS 则使用 ES5 提供的 Object.defineProperty() 方法,监控对数据的操作,从而可以自动触发数据同步。并且,由于是在不同的数据上触发同步,可以精确的将变更发送给绑定的视图,而不是对所有的数据都执行一次检测。

参考:

Vue 双向数据绑定实现

数据与视图的绑定与同步,最终体现在对数据的读写处理过程中,也就是 Object.defineProperty() 定义的数据 set、get 函数中。Vue 中对于的函数为 defineReactive,在精简版实现中,我只保留了一些基本特性:

function defineReactive(obj, key, value) {
    var dep = new Dep()
    Object.defineProperty(obj, key, {
        enumerable: true,
        configurable: true,
        get: function reactiveGetter() {
            if (Dep.target) {
                dep.depend()
            }
            return value
        },
        set: function reactiveSetter(newVal) {
            if (value === newVal) {
                return
            } else {
                value = newVal
                dep.notify()
            }
        }
    })
}

在对数据进行读取时,如果当前有 Watcher(对数据的观察者吧,watcher 会负责将获取的新数据发送给视图),那将该 Watcher 绑定到当前的数据上(dep.depend(),dep 关联当前数据和所有的 watcher 的依赖关系),是一个检查并记录依赖的过程。而在对数据进行赋值时,如果数据发生改变,则通知所有的 watcher(借助 dep.notify())。这样,即便是我们手动改变了数据,框架也能够自动将数据同步到视图。

Paste_Image.png

数据绑定关系的识别过程

Vue 和 AngularJS 中,都是通过在 HTML 中添加指令的方式,将视图元素与数据的绑定关系进行声明。例如:

<form id="test">
  <input type="text" v-model="name">
</form>

以上的 HTML 代码表示该 input 元素与 name 数据进行绑定。在 JS 代码中可以这样进行初始化:

var vm = new Vue({
  el: '#test',
  data: {
    name: 'luobo'
  }
})

代码正确执行后,页面上 input 元素对应的位置会显示上面代码中给出的初始值:luobo。

由于双向数据绑定已经建立,因此:

  • 执行 vm.name = 'mickey' 后,页面上 input 也会更新为显示: mickey
  • 在页面文本框中修改内容为:tang,则通过vm.name 获取的值为:"tang"

那么初始化的过程中,Vue 是如何识别出这种绑定关系的呢?

通过分析源码,在初始化过程中(new Vue() 执行时),主要执行两个步骤:

  • compile
  • link

compile 过程中,对于给定的目标元素进行解析,识别出所有绑定在元素(通过 el 属性传入)上的指令。
link 过程中,建立这些指令与对应数据(通过 data 属性传入初始值)的绑定关系,并以数据的初始值进行渲染。绑定关系建立后,就可以双向同步数据了。

除了基本的双向数据绑定,Vue 还提供了更多的特性和功能,如果只是对双向数据绑定感兴趣,可以看下我的精简版实现:
https://github.com/luobotang/simply-vue
基本是从 Vue 代码中精简、改造得到的,主要保留了 Vue 中与双向数据绑定有关的部分(包括 compile、link 相关代码),指令只保留了 input[type="text"] 和普通文本两种类型,用于演示数据绑定的效果。

相关文章

  • 深入Vue响应式原理

    1.Vue的双向数据绑定 参考 vue的双向绑定原理及实现Vue双向绑定的实现原理Object.definepro...

  • vue 双向数据绑定

    Vue实现数据双向绑定的原理:Object.defineProperty()vue实现数据双向绑定主要是:采用数据...

  • 前端理论面试--VUE

    vue双向绑定的原理(详细链接) VUE实现双向数据绑定的原理就是利用了 Object.definePropert...

  • Vue实现数据双向绑定的原理

    Vue实现数据双向绑定的原理:Object.defineProperty() vue实现数据双向绑定主要是:采用数...

  • 【转】JavaScript的观察者模式(Vue双向绑定原理)

    关于Vue实现数据双向绑定的原理,请点击:Vue实现数据双向绑定的原理原文链接:JavaScript设计模式之观察...

  • vue面试知识点

    vue 数据双向绑定原理 vue实现数据双向绑定原理主要是:采用数据劫持结合发布订阅设计模式的方式,通过对data...

  • Vue双向数据绑定原理

    剖析Vue实现原理 - 如何实现双向绑定mvvm 本文能帮你做什么?1、了解vue的双向数据绑定原理以及核心代码模...

  • 关于双向绑定的问题

    剖析Vue实现原理 - 如何实现双向绑定mvvm 本文能帮你做什么?1、了解vue的双向数据绑定原理以及核心代码模...

  • 前端面试题:VUE

    1. vue的双向数据绑定实现原理? 2. vue如何在组件之间进行传值? 3. vuex和vue的双向数据绑定...

  • vue

    1、vue的双向数据绑定实现原理 2、vue如何在组件之间进行传值 3、vuex和vue的双向数据绑定有什么冲突 ...

网友评论

  • 颩濄璐者:您好大神 ! 我是刚接触vue的菜鸟,最近我在做项目的时候发现一个问题 就是根据v-if同一条件判断元素1和元素2 给元素2绑定了一个click方法 结果为true时 元素1点击也触发了 ? 请问大神这个是什么原因?
    颩濄璐者:@防不胜防__ 大神 受我一拜
    ed578103cd75:Vue 会尽可能高效地渲染元素,通常会复用已有元素而不是从头开始渲染。
    如果元素1和元素2相同(即使部分属性值不同),你都需要添加“key”属性来区分元素1和元素2,
    不然元素2可能会复用元素1渲染,这只是可能的答案,希望有效!
    NARUTO_86:能看懂你的描述我才算是大神吧....很遗憾,看不懂:stuck_out_tongue_winking_eye:
    要不,你还是贴段代码?
  • fc9a3f107025:请问下 我改的这个项目用到的就是input text文本框双向数据绑定,但这个双向可不可以更改下 因为一些数据需要每次打开重新填写,白话说就是每次打开页面 文本框的值为空 不绑定数据库的值,这个基础上可以实现么
    NARUTO_86:@fc9a3f107025 说下我的看法:
    1. 要么不实现为“双向绑定”
    2. 要么开始的时候不提供对应的值
    3. 在框架基础上实现特殊的指令,根据特殊条件判断绝对是否显示值
  • 带三本书:好文章,再问一句,angular的脏检查除了主动调用外,被动调用的原理是轮询么?感觉效率很低的样子
    NARUTO_86:@带三本书 谈不上指点吧,我也只是看些资料学习,最多看一部分源码进行印证。
    “轮询”,难道有一个定时器一直在跑着吗,我想不会这么设计吧。
    能够导致数据变化,要么来自开发者的代码,要么来自用户交互触发。
    来自开发者的代码显然需要自己调用 '$digest'(参考:https://github.com/xufei/blog/issues/10),而用户交互则由框架处理,所以“推测”是在框架的指令这里触发了“脏值检查”。
    “双向绑定”的目的不就是实现数据和UI的同步吗?Angular 只是通过“脏值检测”来识别数据变更,进而同步UI罢了。所以触发“脏值检测”的话就要看是什么情况下导致了数据变更,也就是我理解的上面两种情况。
    带三本书:@luobo_tang 如何检测数据的变化?并把数据的变化展现在view上?除了主动调用脏检测以外?有其他的机制吗?比如说我不主动调用脏检测的api,而是框架轮训自动发现?这方面的机制我不太了解,期望指点一二
    NARUTO_86:@带三本书 “被动调用”不就是 Angular 框架自己执行的嘛,这个合理推测的话就是界面交互产生的事件进一步通过双向数据绑定触发的。
  • 七日夜:很好,学习了
  • 框框之上:请问下,这个数据绑定好像只能绑一个?我在同一个页面下,两个并列的input想绑定到两个并列的span内,一个id设置在大div上,一个id设置在body上,就无法实现,大概是有冲突?
    NARUTO_86:@框框之上 不是很明白你描述的情况。能否贴一下相关的代码?

本文标题:Vue 双向数据绑定原理分析

本文链接:https://www.haomeiwen.com/subject/itkgjttx.html