vue双向数据绑定实现原理学习笔记

参考链接:https://www.cnblogs.com/kidney/p/6052935.html
黄轶的源码解读:https://github.com/DDFE/DDFE-blog/issues/7

一、双向数据绑定和单向数据绑定概念
        双向数据绑定就是在单向数据绑定的基础上给可输入元素(input、textare等)添加了change(input)事件,来动态修改model(js)和 view(视图),在单向数据绑定中,input输入元素中输入的内容可以通过js操作dom动态获取,js中改变的数据也需要再次操作dom反映到视图中。双向数据绑定通过watcher方法自动更新视图中的数据,省去了烦琐的dom操作;
二、访问器属性
  var obj = {}
  // 为obj对象定义一个名为hello的访问器属性
  // 访问器属性是对象中的一种特殊属性,不能直接在对象中定义,只能通defineProperty方法定义
  // 读取或设置访问器属性的值,实际上是调用其内部函数get或set方法
  Object.defineProperty(obj, "hello", {
    get: function() {},
    set: function() {}
  })
  obj.hello // 调用get方法,并返回get方法的返回值
  obj.hello = "123" // 赋值传参,调用set方法,参数是123
  // 访问器属性会被优先访问,即访问器属性会覆盖同名属性
三、双向数据绑定的简化版
var obj = {}
  Object.defineProperty(obj, "hello", {
    get: function() {},
    set: function(newVal) {
      document.getElementById('a').value = newVal
      document.getElementById('b').innerHTML = newVal
    }
  })
  // 模拟watcher
  document.addEventListener('keyup', function(e) {
    obj.hello = e.target.value
  })
四、将vue中的值单向绑定到dom中

1)DocumentFragment文档片断
        可以看做是节点容器,它可以包含多个子节点,将其插入到dom中时,只有它的子节点会插入到目标节点;
        使用DocumentFragment处理节点,速度和性能远远优于直接操作dom;
        vue进行编译时,就是将挂载目标的所有子节点劫持(通过append方法,dom中的所有节点会被自动删除)到DocumentFragment中,处理后再将DocumentFragment整体返回插入挂载目标;

// html代码
<div id="app">
    <input type="text" id="a">
    <span id="b"></span>
  </div>
// js操作
var dom = nodeToFragment(document.getElementById('app'))
  console.log(dom)
  function nodeToFragment(node) {
    var flag = document.createDocumentFragment()
    var child
    while (child = node.firstChild) {
      flag.appendChild(child) // 将子节点劫持到文档片断中
    }
    return flag
  }
  document.getElementById('app').appendChild(dom) // 返回到app中
屏幕快照 2018-07-23 下午4.16.01.png

2)dom编译和数据绑定

// html代码
  <div id="app">
    <input type="text" v-model="text">
    {{ text }}
  </div>
// js代码
// 对dom进行编译,将输入框以及文本节点与data中的数据绑定
  function compile(node, vm) {
    var reg = /\{\{(.*)\}\}/
    // 节点类型为元素
    if (node.nodeType === 1) {
      var attr = node.attributes
      // 解析属性
      for (var i = 0; i < attr.length; i++) {
        if (attr[i].nodeName == 'v-model') {
          var name = attr[i].nodeValue // 获取v-model绑定的属性名
          node.value = vm.data[name] // 将data的值赋给该node
          node.removeAttribute('v-model')
        }
      }
    }
    // 节点类型为text
    if (node.nodeType === 3) {
      if (reg.test(node.nodeValue)) {
        var name = RegExp.$1 // 获取匹配到的字符串
        name = name.trim()
        node.nodeValue = vm.data[name] // 将data的值赋给该node
      }
    }
  }
// 将节点转换为文档片断
  function nodeToFragment(node, vm) {
    var flag = document.createDocumentFragment()
    var child
    while (child = node.firstChild) {
      compile(child, vm)
      flag.appendChild(child) // 将子节点劫持到文档片断中
    }
    return flag
  }
// vue绑定的完整操作
  function Vue(options) {
    this.data = options.data
    var id = options.el
    var dom = nodeToFragment(document.getElementById(id), this)
    // 编译完成后,将dom返回到app中
    document.getElementById(id).appendChild(dom)
  }

  var vm = new Vue({
    el: 'app',
    data: {
      text: 'hello world'
    }
  })

最终结果:


屏幕快照 2018-07-23 下午5.03.33.png
五、实现数据与dom双向绑定

        在输入框中输入数据的时候,首先会触发input或者keyup事件,在相应的事件处理程序中,我们获取输入框的value并赋值给vm实例的text属性,利用defineProperty将data中的text设置为vm的访问器属性,会触发set方法更新属性的值;

// html代码
<div id="app">
    <input type="text" v-model="text">
    {{ text }}
  </div>
// js代码
var obj = {}

  function defineReactive(obj, key, val) {
    Object.defineProperty(obj, key, {
      get: function() {
        return val
      },
      set: function(newVal) {
        if (newVal === val) return
        val = newVal
        console.log(val)
      }
    })
  }

  // watcher
  function observe(obj, vm) {
    Object.keys(obj).forEach(function(key) {
      defineReactive(vm, key, obj[key])
    })
  }

  function Vue(options) {
    this.data = options.data
    var data = this.data
    observe(data, this)
    var id = options.el
    var dom = nodeToFragment(document.getElementById(id), this)
    // 编译完成后,将dom返回到app中
    document.getElementById(id).appendChild(dom)
  }

  // 对dom进行编译,将输入框以及文本节点与data中的数据绑定
  function compile(node, vm) {
    var reg = /\{\{(.*)\}\}/
    // 节点类型为元素
    if (node.nodeType === 1) {
      var attr = node.attributes
      // 解析属性
      for (var i = 0; i < attr.length; i++) {
        if (attr[i].nodeName == 'v-model') {
          var name = attr[i].nodeValue // 获取v-model绑定的属性名
          node.addEventListener('input', function(e) {
            // 给相应的data属性赋值,进而触发该属性的set方法
            vm[name] = e.target.value
          })
          node.value = vm[name] // 将data的值赋给该node
          node.removeAttribute('v-model')
        }
      }
    }
    // 节点类型为text
    if (node.nodeType === 3) {
      if (reg.test(node.nodeValue)) {
        var name = RegExp.$1 // 获取匹配到的字符串
        name = name.trim()
        node.nodeValue = vm[name] // 将data的值赋给该node
      }
    }
  }

  function nodeToFragment(node, vm) {
    var flag = document.createDocumentFragment()
    var child
    while (child = node.firstChild) {
      compile(child, vm)
      flag.appendChild(child) // 将子节点劫持到文档片断中
    }
    return flag
  }

  var vm = new Vue({
    el: 'app',
    data: {
      text: 'hello world'
    }
  })

结果如下:


屏幕快照 2018-07-23 下午5.44.28.png
六、实现数据与dom双向绑定

        text 文本变化了,set方法触发了,使用订阅发布模式将绑定到text的文本节点同步变化,订阅发布模式是一种一对多的关系,即多个观察者同时监听一个主题对象,这个主题对象的状态发生变化时会通知所有观察者对象;
        流程:发布者发出通知=》主题对象收到通知并推送给观察者=》订阅者执行相应操作

 //  一个发布者publisher
  var pub = {
    publish: function() {
      dep.notify()
    }
  }
  // 三个订阅者subscribers
  var sub1 = {
    update: function() {
      console.log(1)
    }
  }
  var sub2 = {
    update: function() {
      console.log(2)
    }
  }
  var sub3 = {
    update: function() {
      console.log(3)
    }
  }

  // 一个主题对象
  function Dep() {
    this.subs = [sub1, sub2, sub3]
  }
  Dep.prototype.notify = function() {
    this.subs.forEach(function(sub) {
      sub.update()
    })
  }
  // 发布者发布消息,主题对象执行notif方法,进而触发订阅者执行update方法
  var dep = new Dep()
  pub.publish() // 1,2,3
七、双向数据绑定完整代码

        监听数据的过程中,会为data中的每一个属性生成一个主题对象dep;
        在编译html过程中,会为每个与数据绑定相关的节点生成一个订阅者watcher,watcher会将自己添加到相应属性的dep中;
        发出通知dep.notify()=>触发订阅者的update方法=>更新视图;

  function defineReactive(obj, key, val) {
    var dep = new Dep()
    Object.defineProperty(obj, key, {
      get: function() {
        // 添加订阅者watcher到主题对象Dep
        if (Dep.target) dep.addSub(Dep.target)
        return val
      },
      set: function(newVal) {
        if (newVal === val) return
        val = newVal
        // 作为发布者发出通知
        dep.notify()
      }
    })
  }

  // watcher
  function observe(obj, vm) {
    Object.keys(obj).forEach(function(key) {
      defineReactive(vm, key, obj[key])
    })
  }

  // 一个主题对象
  function Dep() {
    this.subs = []
  }
  Dep.prototype = {
    addSub: function(sub) {
      this.subs.push(sub)
    },
    notify: function() {
      this.subs.forEach(function(sub) {
        sub.update()
      })
    }
  }

  function Vue(options) {
    this.data = options.data
    var data = this.data
    observe(data, this)
    var id = options.el
    var dom = nodeToFragment(document.getElementById(id), this)
    // 编译完成后,将dom返回到app中
    document.getElementById(id).appendChild(dom)
  }

  // 对dom进行编译,将输入框以及文本节点与data中的数据绑定
  function compile(node, vm) {
    var reg = /\{\{(.*)\}\}/
    // 节点类型为元素
    if (node.nodeType === 1) {
      var attr = node.attributes
      // 解析属性
      for (var i = 0; i < attr.length; i++) {
        if (attr[i].nodeName == 'v-model') {
          var name = attr[i].nodeValue // 获取v-model绑定的属性名
          node.addEventListener('input', function(e) {
            // 给相应的data属性赋值,进而触发该属性的set方法
            vm[name] = e.target.value
          })
          node.value = vm[name] // 将data的值赋给该node
          node.removeAttribute('v-model')
        }
      }
    }
    // 节点类型为text
    if (node.nodeType === 3) {
      if (reg.test(node.nodeValue)) {
        var name = RegExp.$1 // 获取匹配到的字符串
        name = name.trim()
        // node.nodeValue = vm[name] // 将data的值赋给该node
        new Watcher(vm, node, name)
      }
    }
  }

  function Watcher(vm, node, name) {
    Dep.target = this
    this.name = name
    this.node = node
    this.vm = vm
    this.update()
    Dep.target = null
  }

  Watcher.prototype = {
    update: function() {
      this.get()
      this.node.nodeValue = this.value
    },
    // 获取data中的属性值
    get: function() {
      this.value = this.vm[this.name] // 触发相应属性的get
    }
  }

  function nodeToFragment(node, vm) {
    var flag = document.createDocumentFragment()
    var child
    while (child = node.firstChild) {
      compile(child, vm)
      flag.appendChild(child) // 将子节点劫持到文档片断中
    }
    return flag
  }

  var vm = new Vue({
    el: 'app',
    data: {
      text: 'hello world'
    }
  })

结果如下:


屏幕快照 2018-07-23 下午6.42.45.png
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,684评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,143评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,214评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,788评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,796评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,665评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,027评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,679评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,346评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,664评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,766评论 1 331
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,412评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,015评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,974评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,073评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,501评论 2 343

推荐阅读更多精彩内容