vue的简单使用相信大家都很熟练了,使用起来简单。简单但是简单大部分人不知道其内部的原理是怎么样的,今天我们就来一起实现一个简单的简单vue。 实现之前我们得先看一下Object.defineProperty的简单实现,因为vue主要是简单通过数据劫持来实现的,通过get、简单set来完成数据的简单读取和更新。 从上面可以看到通过get获取数据,通过set监听到数据变化执行相应操作,简单还是简单不明白的话可以去看看Object.defineProperty文档。 流程图 上面主要是初始化操作,针对传过来的数据进行处理 上面主要是代理data到最上层,this.xxx的方式直接访问data 同样是使用Object.defineProperty来监听数据,初始化需要订阅的数据。 把需要订阅的数据到push到watcherTask里,等到时候需要更新的时候就可以批量更新数据了。??下面就是; 遍历订阅池,批量更新视图。 这里代码比较多,我们拆分看你就会觉得很简单了 首先我们先遍历el元素下面的所有子节点,node.nodeType === 3 的意思是当前元素是文本节点,node.nodeType === 1 的意思是当前元素是元素节点。因为可能有的是纯文本的形式,云南idc服务商如纯双花括号就是纯文本的文本节点,然后通过判断元素节点是否还存在子节点,如果有的话就递归调用compile方法。下面重头戏来了,我们拆开看: 上面这个首先判断node节点上是否有v-html这种指令,如果存在的话,我们就发布订阅,怎么发布订阅呢?只需要把当前需要订阅的数据push到watcherTask里面,然后到时候在设置值的时候就可以批量更新了,实现双向数据绑定,也就是下面的操作 然后push的值是一个Watcher的实例,首先他new的时候会先执行一次,执行的操作就是去把纯双花括号 -> 1,也就是说把我们写好的模板数据更新到模板视图上。 ***把当前元素属性剔除出去,站群服务器我们用Vue的时候也是看不到这种指令的,不剔除也不影响 至于Watcher是什么,看下面就知道了 之前发布订阅之后走了这里面的操作,意思就是把当前元素如:node.innerHTML = 这是data里面的值、node.value = 这个是表单的数据 那么我们为什么不直接去更新呢,还需要update做什么,不是多此一举吗? 其实update记得吗?我们在订阅池里面需要批量更新,就是通过调用Watcher原型上的update方法。 在线效果地址,大家可以浏览器看一下效果,由于本人太懒了,gif效果图就先不放了,哈哈???? 完整代码已经放到github上了 -> MyVue 【责任编辑:庞桂玉 TEL:(010)68476606】Object.defineProperty()
html代码结构
<div id="wrap"> <p v-html="test"></p> <input type="text" v-model="form"> <input type="text" v-model="form"> <button @click="changeValue">改变值</button> { { form}} </div> js调用
new Vue({ el: #wrap,简单 data:{ form: 这是form的值, test: <strong>我是粗体</strong>, }, methods:{ changeValue(){ console.log(this.form) this.form = 值被我改变了,气不气?简单 } } }) Vue结构
class Vue{ constructor(){ } proxyData(){ } observer(){ } compile(){ } compileText(){ } } class Watcher{ constructor(){ } update(){ } } Vue constructor 构造函数主要是数据的初始化 proxyData 数据代理 observer 劫持监听所有数据 compile 解析dom compileText 解析dom里处理纯双花括号的操作 Watcher 更新视图操作 Vue constructor 初始化
class Vue{ constructor(options = { }){ this.$el = document.querySelector(options.el); let data = this.data = options.data; // 代理data,使其能直接this.xxx的简单方式访问data,正常的云服务器提供商简单话需要this.data.xxx Object.keys(data).forEach((key)=> { this.proxyData(key); }); this.methods = obj.methods // 事件方法 this.watcherTask = { }; // 需要监听的任务列表 this.observer(data); // 初始化劫持监听所有数据 this.compile(this.$el); // 解析dom } } proxyData 代理data
class Vue{ constructor(options = { }){ ...... } proxyData(key){ let that = this; Object.defineProperty(that, key, { configurable: false, enumerable: true, get () { return that.data[key]; }, set (newVal) { that.data[key] = newVal; } }); } } observer 劫持监听
class Vue{ constructor(options = { }){ ...... } proxyData(key){ ...... } observer(data){ let that = this Object.keys(data).forEach(key=>{ let value = data[key] this.watcherTask[key] = [] Object.defineProperty(data,key,{ configurable: false, enumerable: true, get(){ return value }, set(newValue){ if(newValue !== value){ value = newValue that.watcherTask[key].forEach(task => { task.update() }) } } }) }) } } Watcher
that.watcherTask[key].forEach(task => { task.update() }) 效果
完整代码