Object.defineProperty方法
vue.js 是采用数据劫持结合发布者-订阅者模式的方式,通过Object.defineProperty()来劫持各个属性的setter,getter,在数据变动时发布消息给订阅者,触发相应的监听回调。
具体实现过程的代码如下:
1、定义构造函数
function Vue(option){ this.$el = document.querySelector(option.el); //获取挂载节点 this.$data = option.data; this.$methods = option.methods; this.deps = {}; //所有订阅者集合 目标格式(一对多的关系):{msg: [订阅者1, 订阅者2, 订阅者3], info: [订阅者1, 订阅者2]} this.observer(this.$data); //调用观察者 this.compile(this.$el); //调用指令解析器 }
2、定义指令解析器
Vue.prototype.compile = function (el) { let nodes = el.children; //获取挂载节点的子节点 for (var i = 0; i < nodes.length; i++) { var node = nodes[i]; if (node.children.length) { this.compile(node) //递归获取子节点 } if (node.hasAttribute('l-model')) { //当子节点存在l-model指令 let attrVal = node.getAttribute('l-model'); //获取属性值 node.addEventListener('input', (() => { this.deps[attrVal].push(new Watcher(node, "value", this, attrVal)); //添加一个订阅者 let thisNode = node; return () => { this.$data[attrVal] = thisNode.value //更新数据层的数据 } })()) } if (node.hasAttribute('l-html')) { let attrVal = node.getAttribute('l-html'); //获取属性值 this.deps[attrVal].push(new Watcher(node, "innerHTML", this, attrVal)); //添加一个订阅者 } if (node.innerHTML.match(/{{([^\{|\}]+)}}/)) { let attrVal = node.innerHTML.replace(/[{{|}}]/g, ''); //获取插值表达式内容 this.deps[attrVal].push(new Watcher(node, "innerHTML", this, attrVal)); //添加一个订阅者 } if (node.hasAttribute('l-on:click')) { let attrVal = node.getAttribute('l-on:click'); //获取事件触发的方法名 node.addEventListener('click', this.$methods[attrVal].bind(this.$data)); //将this指向this.$data } } }
3、定义观察者
Vue.prototype.observer = function(data){ for(var key in data){ (function(that){ let val = data[key]; //每一个数据的属性值 that.deps[key] = []; //初始化所有订阅者对象{msg: [订阅者], info: []} var watchers = that.deps[key]; Object.defineProperty(data, key, { //数据劫持 get: function(){ return val; }, set: function(newVal){ //设置值(说明有数据更新) if(val !== newVal){ val = newVal; } // 通知订阅者 watchers.forEach(watcher=>{ watcher.update() }) } }) })(this) } }
4、定义订阅者
function Watcher(el, attr, vm, attrVal) { this.el = el; this.attr = attr; this.vm = vm; this.val = attrVal; this.update(); //更新视图 }
5、更新视图
Watcher.prototype.update = function () { this.el[this.attr] = this.vm.$data[this.val] }
以上代码定义在一个Vue.js文件中,在需要使用双向绑定的地方引入即可。
尝试使用一下:
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <meta http-equiv="X-UA-Compatible" content="ie=edge"> <title>Document</title> <script src="/UploadFiles/2021-04-02/vue.js">更多教程点击《Vue.js前端组件学习教程》,欢迎大家学习阅读。
关于vue.js组件的教程,请大家点击专题vue.js组件学习教程进行学习。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
华山资源网 Design By www.eoogi.com
广告合作:本站广告合作请联系QQ:858582 申请时备注:广告合作(否则不回)
免责声明:本站资源来自互联网收集,仅供用于学习和交流,请遵循相关法律法规,本站一切资源不代表本站立场,如有侵权、后门、不妥请联系本站删除!
免责声明:本站资源来自互联网收集,仅供用于学习和交流,请遵循相关法律法规,本站一切资源不代表本站立场,如有侵权、后门、不妥请联系本站删除!
华山资源网 Design By www.eoogi.com
暂无评论...
RTX 5090要首发 性能要翻倍!三星展示GDDR7显存
三星在GTC上展示了专为下一代游戏GPU设计的GDDR7内存。
首次推出的GDDR7内存模块密度为16GB,每个模块容量为2GB。其速度预设为32 Gbps(PAM3),但也可以降至28 Gbps,以提高产量和初始阶段的整体性能和成本效益。
据三星表示,GDDR7内存的能效将提高20%,同时工作电压仅为1.1V,低于标准的1.2V。通过采用更新的封装材料和优化的电路设计,使得在高速运行时的发热量降低,GDDR7的热阻比GDDR6降低了70%。
更新日志
2024年11月16日
2024年11月16日
- 群星《2024好听新歌36》AI调整音效【WAV分轨】
- 梁朝伟.1986-朦胧夜雨裡(华星40经典)【华星】【WAV+CUE】
- 方芳.1996-得意洋洋【中唱】【WAV+CUE】
- 辛欣.2001-放120个心【上海音像】【WAV+CUE】
- 柏菲·万山红《花开原野1》限量开盘母带ORMCD[低速原抓WAV+CUE]
- 柏菲·万山红《花开原野2》限量开盘母带ORMCD[低速原抓WAV+CUE]
- 潘安邦《思念精选集全纪录》5CD[WAV+CUE]
- 杨千嬅《千嬅新唱金牌金曲》金牌娱乐 [WAV+CUE][985M]
- 杨钰莹《依然情深》首版[WAV+CUE][1G]
- 第五街的士高《印度激情版》3CD [WAV+CUE][2.4G]
- 三国志8重制版哪个武将智力高 三国志8重制版智力武将排行一览
- 三国志8重制版哪个武将好 三国志8重制版武将排行一览
- 三国志8重制版武将图像怎么保存 三国志8重制版武将图像设置方法
- 何方.1990-我不是那种人【林杰唱片】【WAV+CUE】
- 张惠妹.1999-妹力新世纪2CD【丰华】【WAV+CUE】