数据绑定

  数据绑定一般就是指的 将数据 展示到 视图上。目前前端的框架都是使用的mvvm模式实现双绑的。大体上有以下几种方式:

  1.  发布订阅
  2. ng的脏检查
  3. 数据劫持

  vue的话采用的是数据劫持和发布订阅相结合的方式。 而数据劫持用的是Object.defineProperty来实现的, 可以通过绑定get和set来在获取和设置数据的时候触发相应的函数。

实现

  所以我们需要一个监听器Observe来监听数据的变化。在数据发生变化时,我们需要一个Watcher订阅者来更新视图,我们还需要一个指令的解析器compile来解析指令和初始化视图。

    • Observe 监听器: 监听数据的变化, 通知订阅者
    • Watcher 订阅者: 收到数据的变化, 更新视图
    • Compile 解析器: 解析指令,初始化模板,绑定订阅者

  Observe

    监听数据的每一个属性, 由于可能会有多个watcher,所以需要一个容器来存储。

    function Sub() {
this.subs = [];
}
Sub.prototype = {
add(sub) {
this.subs.push(sub);
},
trigger() {
this.subs.forEach(sub => {
sub.update();
})
}
};
Sub.target = null; function observe(data) {
if (typeof data !== 'object' || !data) return;
Object.keys(data).forEach(item => {
let val = data[item];
let sub = new Sub();
Object.defineProperty(data, item, {
enumerable: true,
configurable: false,
get() {
if (Sub.target) {
sub.add(Sub.target);
}
return val;
},
set(newVal) {
val = newVal;
sub.trigger();
}
})
})
}

  

  Watcher

    将对应属性的watcher添加到sub容器中, 当属性变化时,执行更新函数。

    function Watcher(vm, prop, callback) {
this.vm = vm;
this.prop = prop;
this.callback = callback;
Sub.target = this;
let val = this.vm.$data[prop];
Sub.target = null;
this.vaule = val;
} Watcher.prototype.update = function () {
let newValue = this.vm.$data[this.prop];
if (this.value !== newValue) {
this.value = newValue;
this.callback.call(this.vm, newValue);
}
}

  

  Compile

    获取到dom中的指令和初始化模板, 添加watcher更新视图。

    function Compile(vm) {
this.vm = vm;
this.el = vm.$el;
this.init();
} Compile.prototype.init = function () {
let fragment = document.createDocumentFragment();
let child = this.el.firstChild;
while(child) {
fragment.append(child);
child = this.el.firstChild;
}
let childNodes = fragment.childNodes;
Array.from(childNodes).forEach(node => {
if (node.nodeType === 1) {
let attrs = node.attributes;
Array.from(attrs).forEach(attr => {
let name = attr.nodeName;
if (name === 'v-model') {
let prop = attr.nodeValue;
let value = this.vm.$data[prop];
node.value = value;
new Watcher(this.vm, prop, val => {
node.value = val;
});
node.addEventListener('input', e => {
let newVal = e.target.value;
if (value !== newVal) {
this.vm.$data[prop] = newVal;
}
})
}
})
} let reg = /\{\{(.*)\}\}/;
let text = node.textContent;
if (reg.test(text)) {
let prop = RegExp.$1;
let val = this.vm.$data[prop];
node.textContent = val;
new Watcher(this.vm, prop, val => {
node.textContent = val;
});
}
})
this.el.appendChild(fragment);
}

到这里, 基本的思路已经实现完毕, 这里只实现了v-model指令。

  最后,结合 Observe Watcher和 Compile, 就可以成为一个完整的mvvm了。

    <div id="app">
<div>{{val}}</div>
<input type="text" id="input" v-model="val">
</div> <script>
function MyVue(options) {
this.$options = options;
this.$el = options.el;
this.$data = options.data;
this.init();
} MyVue.prototype.init = function () {
observe(this.$data);
new Compile(this);
}; new MyVue({
el: document.getElementById('app'),
data: {
val: 123
}
})
</script>

 当然,这只是简单的实现,没考虑细节,主要是学习思路。

 查看效果 (代码直接在页面上)

    

  

  

自己手动实现简单的双向数据绑定 mvvm的更多相关文章

  1. 简单实现双向数据绑定mvvm。

  2. JavaScript实现简单的双向数据绑定

    什么是双向数据绑定 双向数据绑定简单来说就是UI视图(View)与数据(Model)相互绑定在一起,当数据改变之后相应的UI视图也同步改变.反之,当UI视图改变之后相应的数据也同步改变. 双向数据绑定 ...

  3. 五十行javascript代码实现简单的双向数据绑定

    五十行javascript代码实现简单的双向数据绑定 Vue框架想必从事前端开发的同学都使用过,它的双向数据绑定机制能给我们带来很大的方便.今天闲着没事,尝试着实现一下双向数据绑定,接下来给大家分享一 ...

  4. 利用ES6中的Proxy和Reflect 实现简单的双向数据绑定

    利用ES6中的Proxy (代理) 和 Reflect 实现一个简单的双向数据绑定demo. 好像vue3也把 obj.defineProperty()  换成了Proxy+Reflect. 话不多说 ...

  5. 实现双向数据绑定mvvm

    实现双向数据绑定mvvm  

  6. Angular双向数据绑定MVVM以及基本模式分析

    MVVM: angular的MVVM实现的是双向数据绑定,模型从服务器端抓取到数据,将数据通过控制器(controller)传递到视图(view)显示,视图数据发生变化时同样也会影响到模型数据的变化, ...

  7. React简单实现双向数据绑定

    import React, { Component } from 'react' import ReactDOM from 'react-dom' class App extends Componen ...

  8. 原生js简单实现双向数据绑定原理

    根据对象的访问器属性去监听对象属性的变化,访问器属性不能直接在对象中设置,而必须通过 defineProperty() 方法单独定义. 访问器属性的"值"比较特殊,读取或设置访问器 ...

  9. 深入vue源码,了解vue的双向数据绑定原理

    大家都知道vue是一种MVVM开发模式,数据驱动视图的前端框架,并且内部已经实现了双向数据绑定,那么双向数据绑定是怎么实现的呢? 先手动撸一个最最最简单的双向数据绑定 <div> < ...

随机推荐

  1. Python正则表达式就是这么简单【新手必学】

    一前言本篇文章带大家快速入门正则表达式的使用,正则表达式的规则不仅适用python语言,基本大多数编程语言都适用,在日常使用中极为广泛,读者们有必要学好正则表达式.看完这篇文章,读者们要理解什么是正则 ...

  2. SpringMVC 注解配置

    使用注解配置spring  mvc (1)spring mvc的配置文件 <?xml version="1.0" encoding="UTF-8"?> ...

  3. 「SPOJ10707」Count on a tree II

    「SPOJ10707」Count on a tree II 传送门 树上莫队板子题. 锻炼基础,没什么好说的. 参考代码: #include <algorithm> #include &l ...

  4. 如何用python写个人专属群聊提醒小助手?

    前言 大家还记得教会父母玩微信是什么时候吗?父母学会后,我们的生活就发生了「质」的变化,父母也许会吐槽你的微信头像不好,要你换一个头像. 最近 pk哥 又被母后大人吐槽了,原因是亲戚微信群里某个亲戚生 ...

  5. mysql 获取刚插入行id汇总

    mysql 获取刚插入行id汇总 我们在写数据库程序的时候,经常会需要获取某个表中的最大序号数, 一般情况下获取刚插入的数据的id,使用select max(id) from table 是可以的.但 ...

  6. Lesson 12 banks and their customers

    Why is there no risk to the customer when a bank prints the customer's name on his cheques? When any ...

  7. c++类的创建与使用

    c++类的创建与使用 前言: 之前一直对c++的类的创建与使用不太熟悉,有些概念还是有点模糊,借着这次休息的机会整理一下对应是知识点.如有不正确的地方还希望各位读者批评指正. 一.C++中public ...

  8. 网络编程之TCP三次握手,四次断开

    目录 TCP三次握手 1:上图的名词解释 2:TCP三次握手过程 3:为什么不能改成两次握手? TCP三次握手 1:上图的名词解释 SYN:同步序号.它表示建立连接.TCP规定SYN=1时不能携带数据 ...

  9. AndroidQ强势来袭,国产自研系统还有未来吗?

    我的小米8在重启时,屏幕上总会出现那句让人印象深刻的话--"Powered by android".事实上,几乎所有Android手机都会出现这几个单词--国产智能手机也不例外.这 ...

  10. (4)LoraWAN:Physical Message Formats

    四.Physical Message Formats LoRa数据包结构 LoRaTM调制解调器采用隐式和显式两种数据包格式.其中,显式数据包的报头较短,主要 包含字节数.编码率及是否在数据包中使用循 ...