MVVM 双向绑定的实现代码

网友投稿 234 2023-01-28


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.setAttripGCkybute('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

}

}

// 定义对象属性设置pGCky劫持

// 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.vapGCkylue = '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 数据绑定的元素,而不会去扫描所有元素。而在脏数据检测中,它去找到了所有绑定的元素,然后判断数据是否发生变化,这种方式只有一定的性能开销的。

参考

《现代前端技术解析》

代码下载:https://github.com/OreChou/twowaybinding


版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:Vue 获取数组键名的方法
下一篇:如何让电脑连接共享文件夹(电脑共享文件怎么连接)
相关文章

 发表评论

暂时没有评论,来抢沙发吧~