MVVM 双向绑定的实现

这篇文章主要记录学习 JS 双向绑定过程中的一些概念与具体的实现

MVVM 具体概念

MVVM 中有一些概念是通用的,具体如下

Directive (指令)

自定义的执行函数,例如 Vue 中的 v-click、v-bind 等。这些函数封装了 DOM 的一些基本可复用函数API。

Filter (过滤器)

用户希望对传入的初始数据进行处理,然后将处理结果交给 Directive 或者下一个 Filter。例如:v-bind="time | formatTime"。formatTime 是将 time 转换成指定格式的 Filter 函数。

表达式

类似前端普通的页面模板表达式,作用是控制页面内容安装具体的条件显示。例如:if...else 等

ViewModel

传入的 Model 数据在内存中存放,提供一些基本的操作 API 给开发者,使其能够对数据进行读取与修改

双向绑定(数据变更检测)

View 层的变化改变 Model:通过给元素添加 onchange 事件来触发对 Model 数据进行修改

Model 层的变化改变 View:

  • 手动触发绑定

  • 脏数据检测

  • 对象劫持

  • Proxy

实现方式

手动触发绑定

即 Model 对象改变之后,需要显示的去触发 View 的更新

首先编写 HTML 页面


  Two way binding

编写实现 MVVM 的 代码

// Manual trigger
let elems = [document.getElementById('el'), document.getElementById('input')]
// 数据 Model
let data = {
  value: 'hello'
}

// 定义 Directive
let directive = {
  text: function(text) {
    this.innerHTML = text
  },
  value: function(value) {
    this.setAttribute('value', value)
    this.value = value
  }
}

// 扫描所有的元素
function scan() {
  // 扫描带指令的节点属性
  for (let elem of elems) {
    elem.directive = []
    for (let attr of elem.attributes) {
      if (attr.nodeName.indexOf('q-') >= 0) {
        directive[attr.nodeName.slice(2)].call(elem, data[attr.nodeValue])
        elem.directive.push(attr.nodeName.slice(2))
      }
    }
  }
}

// ViewModel 更新函数
function ViewModelSet(key, value) {
  // 修改数据对象后
  data[key] = value
  // 手动地去触发 View 的修改
  scan()
}

// View 绑定监听
elems[1].addEventListener('keyup', function(e) {
  ViewModelSet('value', e.target.value)
}, false)

// -------- 程序执行 -------
scan()
setTimeout(() => {
  ViewModelSet('value', 'hello world')
}, 1000);

数据劫持

数据劫持是目前比较广泛的方式,Vue 的双向绑定就是通过数据劫持实现。实现方式是通过 Object.defineProperty 和 Object.defineProperies 方法对 Model 对象的 get 和 set 函数进行监听。当有数据读取或赋值操作时,扫描(或者通知)对应的元素执行 Directive 函数,实现 View 的刷新。

HTML 的代码不变,js 代码如下

// Hijacking
let elems = [document.getElementById('el'), document.getElementById('input')]
let data = {
  value: 'hello'
}

// 定义 Directive
let directive = {
  text: function(text) {
    this.innerHTML = text
  },
  value: function(value) {
    this.setAttribute('value', value)
    this.value = value
  }
}

// 定义对象属性设置劫持
// obj: 指定的 Model 数据对象
// propName: 指定的属性名称
function defineGetAndSet(obj, propName) {
  let bValue
  // 使用 Object.defineProperty 做数据劫持
  Object.defineProperty(obj, propName, {
    get: function() {
      return bValue
    },
    set: function(value) {
      bValue = value
      // 在 vue 中,这里不会去扫描所有的元素,而是通过订阅发布模式,通知那些订阅了该数据的 view 进行更新
      scan()
    },
    enumerable: true,
    configurable: true
  })
}

// View 绑定监听
elems[1].addEventListener('keyup', function(e) {
  data.value = e.target.value
}, false)

// 扫描所有的元素
function scan() {
  // 扫描带指令的节点属性
  for (let elem of elems) {
    elem.directive = []
    for (let attr of elem.attributes) {
      if (attr.nodeName.indexOf('q-') >= 0) {
        directive[attr.nodeName.slice(2)].call(elem, data[attr.nodeValue])
        elem.directive.push(attr.nodeName.slice(2))
      }
    }
  }
}

// -------- 程序执行 -------
scan()
defineGetAndSet(data, 'value')
setTimeout(() => {
  // 这里为数据设置新值之后,在 set 方法中会去更新 view
  data.value = 'Hello world'
}, 1000);

基于 Proxy 的实现

Proxy 是 ES6 中的新特性。可以在已有的对象基础上定义一个新对象,并重新定义对象原型上的方法。例如 get 和 set 方法。

// Hijacking
let elems = [document.getElementById('el'), document.getElementById('input')]

// 定义 Directive
let directive = {
  text: function(text) {
    this.innerHTML = text
  },
  value: function(value) {
    this.setAttribute('value', value)
    this.value = value
  }
}

// 设置对象的代理
let data = new Proxy({}, {
  get: function(target, key, receiver) {
    return target.value
  },
  set: function (target, key, value, receiver) { 
    target.value = value
    scan()
    return target.value
  }
})

// View 绑定监听
elems[1].addEventListener('keyup', function(e) {
  data.value = e.target.value
}, false)

// 扫描所有的元素
function scan() {
  // 扫描带指令的节点属性
  for (let elem of elems) {
    elem.directive = []
    for (let attr of elem.attributes) {
      if (attr.nodeName.indexOf('q-') >= 0) {
        directive[attr.nodeName.slice(2)].call(elem, data[attr.nodeValue])
        elem.directive.push(attr.nodeName.slice(2))
      }
    }
  }
}

// -------- 程序执行 -------
data['value'] = 'Hello'
scan()
setTimeout(() => {
  data.value = 'Hello world'
}, 1000);

脏数据监测

基本原理是在 Model 对象的属性值发生变化的时候找到与该属性值相关的所有元素,然后判断数据是否发生变化,若变化则更新 View。

编写页面代码如下:


  Two way binding

js 代码如下:

// Dirty detection
let elems = [document.getElementById('el'), document.getElementById('input')]
let data = {
  value: 'hello'
}

// 定义 Directive
let directive = {
  text: function(text) {
    this.innerHTML = text
  },
  value: function(value) {
    this.setAttribute('value', value)
    this.value = value
  }
}

// 脏数据循环检测
function digest(elems) {
  for (let elem of elems) {
    if (elem.directive === undefined) {
      elem.directive = {}
    }
    for (let attr of elem.attributes) {
      if (attr.nodeName.indexOf('q-event') >= 0) {
        let dataKey = elem.getAttribute('q-bind') || undefined
        // 进行脏数据检测,如果数据改变,则重新执行命令
        if (elem.directive[attr.nodeValue] !== data[dataKey]) {
          directive[attr.nodeValue].call(elem, data[dataKey])
          elem.directive[attr.nodeValue] = data[dataKey]
        }
      }
    }
  }
}

// 数据监听
function $digest(value) {
  let list = document.querySelectorAll('[q-bind=' + value + ']')
  digest(list)
}

// View 绑定监听
elems[1].addEventListener('keyup', function(e) {
  data.value = e.target.value
  $digest(e.target.getAttribute('q-bind'))
}, false)

// -------- 程序执行 -------
$digest('value')
setTimeout(() => {
  data.value = "Hello world"
  $digest('value')
}, 1000);

总结

上面只是简单地实现了双向绑定,但实际上一个完整的 MVVM 框架要考虑很多东西。在上面的实现中数据劫持的方法更新View 是使用了 Scan 函数,但实际的实现中(比如 Vue)是使用了发布订阅的模式。它只会去更新那些与该 Model 数据绑定的元素,而不会去扫描所有元素。而在脏数据检测中,它去找到了所有绑定的元素,然后判断数据是否发生变化,这种方式只有一定的性能开销的。

参考

《现代前端技术解析》

代码下载

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

推荐阅读更多精彩内容