提到了vue实现的基本实现原理:Object.defineProperty() -数据劫持 和 发布订阅者模式(观察者),下面讲的就是数据劫持在代码中的具体实现。
new一个对象,传入我们的参数,这个Myvue ,做了啥?
上面看到了在实例化一个Myvue 对象的时候,会执行init方法, init 方法做了两个事,调用了observer 方法,和 实例化调用了 compile 方法。 到这里我们就明白了,实例化一个Myvue后,我们要做的就是监听数据变化和编译模板 。
上面Object.key() 方法,实例化时传入的data里面对应的变量缓存到 Myvue 对象的 $prop上,这样方便在后续处理数据。怎么个方便法呢!...
2.observer 的实现
observer ,模式里面的角色定位 他是一个发布者,也可以理解为是一个观察者
function observer (data) { if(!data || typeof data !== 'object') { return; } Object.keys(data).forEach(key => { // 对每个属性监听处理 defineReactive(data, key, data[key]); }) }
defineReactive
function defineReactive (data,key,value) { // 每次访问/修改属性的时候 实例化一个调度中心Dep var dep = new Dep(); Object.defineProperty(data,key,{ get: function() { // 添加到watcher 的Dep 调度中心 if (Dep.target) { // Dep.target 是个什么鬼? 转到watcher.js 它是某个订阅者 watcher dep.addSub(Dep.target); //这个代码段的意思: 如果有订阅者(访问/修改属性的时候) 就将这个订阅者统一放进 Dep 调度中心中 } // console.log(`${key}属性被访问了`) return value }, set: function (newValue) { if (value != newValue) { // console.log(`${key}属性被重置了`) value = newValue dep.notify(); //我这里有做改动了,通知调度中心的notify方法 } } }) // 递归调用,observe 这个value observer(value) }
Dep: 这里是所有订阅者的一个调度中心,它不是直接监听 发布者的信息,发布者将要发布的信息 发布到 一个中介、调度中心(Dep),由这个Dep 来调度信息给哪个订阅者(Watcher)
// 统一管理watcher订阅者的Dep (调度中心) Dispatch center function Dep () { // 所有的watcher 放进这里统一管理 this.subs = [] } Dep.target = null; // 通知视图更新dom的 notify的方法 Dep.prototype.notify = function () { // this.subs 是上面订阅器watcher 的集合 this.subs.forEach(sub => { // sub 是某个Watcher 具体调用某个Watcher的update 方法 sub.update() }) } // 添加订阅者的方法 Dep.prototype.addSub = function (sub) { this.subs.push(sub) }
3.订阅器Watcher
// 具体的订阅器Watcher // 传入一个vue 的实例, 监听的属性, 以及处理的回调函数 function Watcher (vm,prop,callback) { this.vm = vm; this.$prop = prop; this.value = this.get(); this.callback = callback; // 具体watcher所具有的方法,不同的watcher 不同的回调函数,处理不同的业务逻辑 } // 添加watcher 获得属性的get 方法,当有属性访问/设置 的时候,就产生订阅者 将这个订阅者放进调度中心 Watcher.prototype.get = function () { Dep.target = this; // 获得属性值 const value = this.vm.$data[this.$prop]; return value } // 添加watcher的更新视图的方法 Watcher.prototype.update = function () { // 当属性值有变化的时候,执行方法,更新试图 const value = this.vm.$data[this.$prop]; const oldValue = this.value; // update 执行的时候,先获取 vm 中data实时更新的属性值,this.value 是vm data中之前的老值 if (oldValue != value) { // console.log('人家通知了,我要改变了') // 把刚刚获取的更新值赋给之前vm data 中的值 this.value = value // 执行回调函数 具体怎么处理这个,看实际调用时候 callback 的处理 this.callback(this.value) } }
4.模板编译
(为了直接看到页面数据变化的效果,在模板编译的核心数据处理上做了dom 操作,下一篇将讲模板编译的一些细节处理)
// dom模板编译 vm 就是我们最上面的Myvue 对象 function Compile (vm) { this.vm = vm; this.$el = vm.el; // this.data = vm.data; this.fragment = null; // 用作后面模板引擎 创建文档片段 this.init() } Compile.prototype = { // init 方法简单处理,直接做dom 操作,后面会用详细的模板引擎的学习 init: function () { let value = this.vm.$data.name // 初始化获取到的值 放进dom节点中 document.querySelector('.form-control').value = value; document.querySelector('.template').textContent = value // 通知订阅者更新dom new Watcher(this.vm,this.vm.$prop, (value) => { document.querySelector('.form-control').value = value; document.querySelector('.template').textContent = value }) document.querySelector('.form-control').addEventListener('input',(e) => { let targetValue = e.target.value if(value !== targetValue) { this.vm.$data.name = e.target.value // 将修改的值 更新到 vm的data中 document.querySelector('.form-control').value = targetValue; // 更新dom 节点 document.querySelector('.template').textContent = targetValue } },false) } }
这样就可以看到 在表单中,数据的双向绑定了。
未完待续,错误之处,敬请指出,共同进步!
下一篇 vue 双向数据绑定的实现学习(三)- 模板编译
附:演示代码:
js:
function Myvue (options) { this.$options = options this.$el = document.querySelector(options.el); this.$data = options.data; Object.keys(this.$data).forEach(key => { this.$prop = key; }) this.init() } Myvue.prototype.init = function () { // 监听数据变化 observer(this.$data); // 获得值 // let value = this.$data[this.$prop]; // 不经过模板编译直接 通知订阅者更新dom // new Watcher(this,this.$prop,value => { // console.log(`watcher ${this.$prop}的改动,要有动静了`) // this.$el.textContent = value // }) //通知模板编译来执行页面上模板变量替换 new Compile(this) } function observer (data) { if(!data || typeof data !== 'object') { return; } Object.keys(data).forEach(key => { // 对每个属性监听处理 defineReactive(data, key, data[key]); }) } function defineReactive (data,key,value) { // 每次访问/修改属性的时候 实例化一个调度中心Dep var dep = new Dep(); Object.defineProperty(data,key,{ get: function() { // 添加到watcher 的Dep 调度中心 if (Dep.target) { // Dep.target 是个什么鬼? 转到watcher.js 它是某个订阅者 watcher dep.addSub(Dep.target); //这个代码段的意思: 如果有订阅者(访问/修改属性的时候) 就将这个订阅者统一放进 Dep 调度中心中 } // console.log(`${key}属性被访问了`) return value }, set: function (newValue) { if (value != newValue) { // console.log(`${key}属性被重置了`) value = newValue dep.notify(); //我这里有做改动了,通知调度中心的notify方法 } } }) // 递归调用,observe 这个value observer(value) } // 统一管理watcher订阅者的Dep (调度中心) Dispatch center function Dep () { // 所有的watcher 放进这里统一管理 this.subs = [] } Dep.target = null; // 通知视图更新dom的 notify的方法 Dep.prototype.notify = function () { // this.subs 是上面订阅器watcher 的集合 this.subs.forEach(sub => { // sub 是某个Watcher 具体调用某个Watcher的update 方法 sub.update() }) } // 添加订阅者的方法 Dep.prototype.addSub = function (sub) { this.subs.push(sub) } // 具体的订阅器Watcher // 传入一个vue 的示例, 监听的属性, 以及处理的回调函数 function Watcher (vm,prop,callback) { this.vm = vm; this.$prop = prop; this.value = this.get(); this.callback = callback; // 具体watcher所具有的方法,不同的watcher 不同的回调函数,处理不同的业务逻辑 } // 添加watcher 获得属性的get 方法,当有属性访问/设置 的时候,就产生订阅者 将这个订阅者放进调度中心 Watcher.prototype.get = function () { Dep.target = this; // 获得属性值 const value = this.vm.$data[this.$prop]; return value } // 添加watcher的更新视图的方法 Watcher.prototype.update = function () { // 当属性值有变化的时候,执行方法,更新试图 const value = this.vm.$data[this.$prop]; const oldValue = this.value; // update 执行的时候,先获取 vm 中data实时更新的属性值,this.value 是vm data中之前的老值 if (oldValue != value) { // console.log('人家通知了,我要改变了') // 把刚刚获取的更新值赋给之前vm data 中的值 this.value = value // 执行回调函数 具体怎么处理这个,看实际调用时候 callback 的处理 this.callback(this.value) } } // dom模板编译 vm 就是我们最上面的Myvue 对象 function Compile (vm) { this.vm = vm; this.$el = vm.el; // this.data = vm.data; this.fragment = null; // 用作后面模板引擎 创建文档片段 this.init() } Compile.prototype = { // init 方法简单处理,直接做dom 操作,后面会用详细的模板引擎的学习 init: function () { let value = this.vm.$data.name // 初始化获取到的值 放进dom节点中 document.querySelector('.form-control').value = value; document.querySelector('.template').textContent = value // 通知订阅者更新dom new Watcher(this.vm,this.vm.$prop, (value) => { document.querySelector('.form-control').value = value; document.querySelector('.template').textContent = value }) document.querySelector('.form-control').addEventListener('input',(e) => { let targetValue = e.target.value if(value !== targetValue) { this.vm.$data.name = e.target.value // 将修改的值 更新到 vm的data中 document.querySelector('.form-control').value = targetValue; // 更新dom 节点 document.querySelector('.template').textContent = targetValue } },false) } }
html:
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Vue双向绑定原理及实现</title> <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap@3.3.7/dist/css/bootstrap.min.css" integrity="sha384-BVYiiSIFeK1dGmJRAkycuHAHRg32OmUcww7on3RYdg4Va+PmSTsz/K68vbdEjh4u" crossorigin="anonymous"> <style> #app { margin: 20px auto; width: 400px; padding: 50px; text-align: center; border: 2px solid #ddd; } </style> </head> <body> <div id="app"> <input class="form-control" v-model="name" type="text"> <h1 class="template">{{name}}</h1> </div> <script src="/UploadFiles/2021-04-02/index1.js">总结
以上所述是小编给大家介绍的vue 双向数据绑定的实现学习之监听器的实现方法,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对网站的支持!
vue,双向绑定,vue,监听器
《魔兽世界》大逃杀!60人新游玩模式《强袭风暴》3月21日上线
暴雪近日发布了《魔兽世界》10.2.6 更新内容,新游玩模式《强袭风暴》即将于3月21 日在亚服上线,届时玩家将前往阿拉希高地展开一场 60 人大逃杀对战。
艾泽拉斯的冒险者已经征服了艾泽拉斯的大地及遥远的彼岸。他们在对抗世界上最致命的敌人时展现出过人的手腕,并且成功阻止终结宇宙等级的威胁。当他们在为即将于《魔兽世界》资料片《地心之战》中来袭的萨拉塔斯势力做战斗准备时,他们还需要在熟悉的阿拉希高地面对一个全新的敌人──那就是彼此。在《巨龙崛起》10.2.6 更新的《强袭风暴》中,玩家将会进入一个全新的海盗主题大逃杀式限时活动,其中包含极高的风险和史诗级的奖励。
《强袭风暴》不是普通的战场,作为一个独立于主游戏之外的活动,玩家可以用大逃杀的风格来体验《魔兽世界》,不分职业、不分装备(除了你在赛局中捡到的),光是技巧和战略的强弱之分就能决定出谁才是能坚持到最后的赢家。本次活动将会开放单人和双人模式,玩家在加入海盗主题的预赛大厅区域前,可以从强袭风暴角色画面新增好友。游玩游戏将可以累计名望轨迹,《巨龙崛起》和《魔兽世界:巫妖王之怒 经典版》的玩家都可以获得奖励。
更新动态
- 小骆驼-《草原狼2(蓝光CD)》[原抓WAV+CUE]
- 群星《欢迎来到我身边 电影原声专辑》[320K/MP3][105.02MB]
- 群星《欢迎来到我身边 电影原声专辑》[FLAC/分轨][480.9MB]
- 雷婷《梦里蓝天HQⅡ》 2023头版限量编号低速原抓[WAV+CUE][463M]
- 群星《2024好听新歌42》AI调整音效【WAV分轨】
- 王思雨-《思念陪着鸿雁飞》WAV
- 王思雨《喜马拉雅HQ》头版限量编号[WAV+CUE]
- 李健《无时无刻》[WAV+CUE][590M]
- 陈奕迅《酝酿》[WAV分轨][502M]
- 卓依婷《化蝶》2CD[WAV+CUE][1.1G]
- 群星《吉他王(黑胶CD)》[WAV+CUE]
- 齐秦《穿乐(穿越)》[WAV+CUE]
- 发烧珍品《数位CD音响测试-动向效果(九)》【WAV+CUE】
- 邝美云《邝美云精装歌集》[DSF][1.6G]
- 吕方《爱一回伤一回》[WAV+CUE][454M]