文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Vue2.x的双向绑定原理及实现

2024-12-03 08:00

关注

https://jsrun.net/RMIKp/embed...

MVVM 框架主要包含两个方面,数据变化更新视图,视图变化更新数据。

视图变化更新数据,如果是像 input 这种标签,可以使用 oninput 事件..

数据变化更新视图可以使用 Object.definProperty() 的 set 方法可以检测数据变化,当数据改变就会触发这个函数,然后更新视图。

实现过程

我们知道了如何实现双向绑定了,首先要对数据进行劫持监听,所以我们需要设置一个 Observer 函数,用来监听所有属性的变化。

如果属性发生了变化,那就要告诉订阅者 watcher 看是否需要更新数据,如果订阅者有多个,则需要一个 Dep 来收集这些订阅者,然后在监听器 observer 和 watcher 之间进行统一管理。

还需要一个指令解析器 compile,对需要监听的节点和属性进行扫描和解析。

因此,流程大概是这样的:

  1.  实现一个监听器 Observer,用来劫持并监听所有属性,如果发生变动,则通知订阅者。
  2.  实现一个订阅者 Watcher,当接到属性变化的通知时,执行对应的函数,然后更新视图,使用 Dep 来收集这些 Watcher。
  3.  实现一个解析器 Compile,用于扫描和解析的节点的相关指令,并根据初始化模板以及初始化相应的订阅器。

显示一个 Observer

Observer 是一个数据监听器,核心方法是利用 Object.defineProperty() 通过递归的方式对所有属性都添加 setter、getter 方法进行监听。 

  1. var library = {  
  2.   book1: {  
  3.     name: "",  
  4.   },  
  5.   book2: "",  
  6. };  
  7. observe(library);  
  8. library.book1.name = "vue权威指南"; // 属性name已经被监听了,现在值为:“vue权威指南”  
  9. library.book2 = "没有此书籍"; // 属性book2已经被监听了,现在值为:“没有此书籍”  
  10. // 为数据添加检测  
  11. function defineReactive(data, key, val) {  
  12.   observe(val); // 递归遍历所有子属性  
  13.   let dep = new Dep(); // 新建一个dep  
  14.   Object.defineProperty(data, key, {  
  15.     enumerable: true,  
  16.     configurable: true,  
  17.     get: function() {  
  18.       if (Dep.target) {  
  19.         // 判断是否需要添加订阅者,仅第一次需要添加,之后就不用了,详细看Watcher函数  
  20.         dep.addSub(Dep.target); // 添加一个订阅者  
  21.       } 
  22.       return val;  
  23.     },  
  24.     set: function(newVal) {  
  25.       if (val == newVal) return; // 如果值未发生改变就return  
  26.       val = newVal 
  27.       console.log(  
  28.         "属性" + key + "已经被监听了,现在值为:“" + newVal.toString() + "”"  
  29.       );  
  30.       dep.notify(); // 如果数据发生变化,就通知所有的订阅者。  
  31.     },  
  32.   }); 
  33.  
  34. // 监听对象的所有属性  
  35. function observe(data) {  
  36.   if (!data || typeof data !== "object") {  
  37.     return; // 如果不是对象就return 
  38.   }  
  39.   Object.keys(data).forEach(function(key) {  
  40.     defineReactive(data, key, data[key]);  
  41.   });  
  42.  
  43. // Dep 负责收集订阅者,当属性发生变化时,触发更新函数。  
  44. function Dep() {  
  45.   this.subs = {};  
  46.  
  47. Dep.prototype = {  
  48.   addSub: function(sub) {  
  49.     this.subs.push(sub);  
  50.   },  
  51.   notify: function() {  
  52.     this.subs.forEach((sub) => sub.update());  
  53.   },  
  54. }; 

思路分析中,需要有一个可以容纳订阅者消息订阅器 Dep,用于收集订阅者,在属性发生变化时执行对应的更新函数。

从代码上看,将订阅器 Dep 添加在 getter 里,是为了让 Watcher 初始化时触发,,因此,需要判断是否需要订阅者。

在 setter 中,如果有数据发生变化,则通知所有的订阅者,然后订阅者就会更新对应的函数。

到此为止,一个比较完整的 Observer 就完成了,接下来开始设计 Watcher.

实现 Watcher

订阅者 Watcher 需要在初始化的时候将自己添加到订阅器 Dep 中,我们已经知道监听器 Observer 是在 get 时执行的 Watcher 操作,所以只需要在 Watcher 初始化的时候触发对应的 get 函数去添加对应的订阅者操作即可。

那给如何触发 get 呢?因为我们已经设置了 Object.defineProperty(),所以只需要获取对应的属性值就可以触发了。

我们只需要在订阅者 Watcher 初始化的时候,在 Dep.target 上缓存下订阅者,添加成功之后在将其去掉就可以了。 

  1. function Watcher(vm, exp, cb) {  
  2.   this.cb = cb;  
  3.   this.vm = vm;  
  4.   this.exp = exp;  
  5.   thisthis.value = this.get(); // 将自己添加到订阅器的操作  
  6.  
  7. Watcher.prototype = {  
  8.   update: function() {  
  9.     this.run();  
  10.   },  
  11.   run: function() {  
  12.     var value = this.vm.data[this.exp];  
  13.     var oldVal = this.value; 
  14.     if (value !== oldVal) {  
  15.       this.value = value;  
  16.       this.cb.call(this.vm, value, oldVal);  
  17.     }  
  18.   },  
  19.   get: function() {  
  20.     Dep.target = this; // 缓存自己,用于判断是否添加watcher。  
  21.     var value = this.vm.data[this.exp]; // 强制执行监听器里的get函数  
  22.     Dep.target = null; // 释放自己  
  23.     return value;  
  24.   },  
  25. }; 

到此为止, 简单的额 Watcher 设计完毕,然后将 Observer 和 Watcher 关联起来,就可以实现一个简单的的双向绑定了。

因为还没有设计解析器 Compile,所以可以先将模板数据写死。

将代码转化为 ES6 构造函数的写法,预览试试。

https://jsrun.net/8SIKp/embed...

这段代码因为没有实现编译器而是直接传入了所绑定的变量,我们只在一个节点上设置一个数据(name)进行绑定,然后在页面上进行 new MyVue,就可以实现双向绑定了。

并两秒后进行值得改变,可以看到,页面也发生了变化。 

  1. // MyVue  
  2. proxyKeys(key) {  
  3.     var self = this 
  4.     Object.defineProperty(this, key, {  
  5.         enumerable: false,  
  6.         configurable: true,  
  7.         get: function proxyGetter() {  
  8.             return self.data[key];  
  9.         },  
  10.         set: function proxySetter(newVal) {  
  11.             self.data[key] = newVal;  
  12.         }  
  13.     });  

上面这段代码的作用是将 this.data 的 key 代理到 this 上,使得我可以方便的使用 this.xx 就可以取到 this.data.xx。

实现 Compile

虽然上面实现了双向数据绑定,但是整个过程都没有解析 DOM 节店,而是固定替换的,所以接下来要实现一个解析器来做数据的解析和绑定工作。

解析器 compile 的实现步骤:

  1.  解析模板指令,并替换模板数据,初始化视图。
  2.  将模板指定对应的节点绑定对应的更新函数,初始化相应的订阅器。

为了解析模板,首先需要解析 DOM 数据,然后对含有 DOM 元素上的对应指令进行处理,因此整个 DOM 操作较为频繁,可以新建一个 fragment 片段,将需要的解析的 DOM 存入 fragment 片段中在进行处理。 

  1. function nodeToFragment(el) {  
  2.   var fragment = document.createDocumentFragment();  
  3.   var child = el.firstChild;  
  4.   while (child) {  
  5.     // 将Dom元素移入fragment中  
  6.     fragment.appendChild(child);  
  7.     child = el.firstChild;  
  8.   }  
  9.   return fragment;  

接下来需要遍历各个节点,对含有相关指令和模板语法的节点进行特殊处理,先进行最简单模板语法处理,使用正则解析“{{变量}}”这种形式的语法。 

  1. function compileElement (el) {  
  2.     var childNodes = el.childNodes;  
  3.     var self = this 
  4.     [].slice.call(childNodes).forEach(function(node) {  
  5.         var reg = /\{\{(.*)\}\}/; // 匹配{{xx}}  
  6.         var text = node.textContent;  
  7.         if (self.isTextNode(node) && reg.test(text)) {  // 判断是否是符合这种形式{{}}的指令  
  8.             self.compileText(node, reg.exec(text)[1]);  
  9.         }  
  10.         if (node.childNodes && node.childNodes.length) {  
  11.             self.compileElement(node);  // 继续递归遍历子节点  
  12.         }  
  13.     });  
  14. },  
  15. function compileText (node, exp) {  
  16.     var self = this 
  17.     var initText = this.vm[exp];  
  18.     updateText(node, initText);  // 将初始化的数据初始化到视图中  
  19.     new Watcher(this.vm, exp, function (value) {  // 生成订阅器并绑定更新函数  
  20.         self.updateText(node, value);  
  21.     });  
  22. },  
  23. function updateText (node, value) {  
  24.     node.textContent = typeof value == 'undefined' ? '' : value;  

获取到最外层的节点后,调用 compileElement 函数,对所有的子节点进行判断,如果节点是文本节点切匹配{{}}这种形式的指令,则进行编译处理,初始化对应的参数。

然后需要对当前参数生成一个对应的更新函数订阅器,在数据发生变化时更新对应的 DOM。

这样就完成了解析、初始化、编译三个过程了。

接下来改造一个 myVue 就可以使用模板变量进行双向数据绑定了。

https://jsrun.net/K4IKp/embed...

添加解析事件

添加完 compile 之后,一个数据双向绑定就基本完成了,接下来就是在 Compile 中添加更多指令的解析编译,比如 v-model、v-on、v-bind 等。

添加一个 v-model 和 v-on 解析: 

  1. function compile(node) {  
  2.   var nodenodeAttrs = node.attributes; 
  3.   var self = this 
  4.   Array.prototype.forEach.call(nodeAttrs, function(attr) {  
  5.     var attrattrName = attr.name; 
  6.     if (isDirective(attrName)) {  
  7.       var exp = attr.value;  
  8.       var dir = attrName.substring(2);  
  9.       if (isEventDirective(dir)) {  
  10.         // 事件指令  
  11.         self.compileEvent(node, self.vm, exp, dir);  
  12.       } else { 
  13.         // v-model 指令  
  14.         self.compileModel(node, self.vm, exp, dir);  
  15.       }  
  16.       node.removeAttribute(attrName); // 解析完毕,移除属性  
  17.     }  
  18.   });  
  19.  
  20. // v-指令解析  
  21. function isDirective(attr) {  
  22.   return attr.indexOf("v-") == 0;  
  23.  
  24. // on: 指令解析  
  25. function isEventDirective(dir) {  
  26.   return dir.indexOf("on:") === 0;  

上面的 compile 函数是用于遍历当前 dom 的所有节点属性,然后判断属性是否是指令属性,如果是在做对应的处理(事件就去监听事件、数据就去监听数据..)

完整版 myVue

在 MyVue 中添加 mounted 方法,在所有操作都做完时执行。 

  1. class MyVue {  
  2.   constructor(options) {  
  3.     var self = this 
  4.     this.data = options.data;  
  5.     this.methods = options.methods;
  6.      Object.keys(this.data).forEach(function(key) {  
  7.       self.proxyKeys(key);  
  8.     });  
  9.     observe(this.data);  
  10.     new Compile(options.el, this);  
  11.     options.mounted.call(this); // 所有事情处理好后执行mounted函数  
  12.   }  
  13.   proxyKeys(key) {  
  14.     // 将this.data属性代理到this上  
  15.     var self = this 
  16.     Object.defineProperty(this, key, {  
  17.       enumerable: false,  
  18.       configurable: true,  
  19.       get: function getter() {  
  20.         return self.data[key];  
  21.       },  
  22.       set: function setter(newVal) {  
  23.         self.data[key] = newVal;  
  24.       },  
  25.     });  
  26.   }  

然后就可以测试使用了。

https://jsrun.net/Y4IKp/embed...

总结一下流程,回头在哪看一遍这个图,是不是清楚很多了。

可以查看的代码地址:Vue2.x 的双向绑定原理及实现 

 

来源:segmentfault内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯