浅谈Vue 数据响应式原理


Posted in Javascript onMay 07, 2018

前言

Vue的数据响应主要是依赖了Object.defineProperty(),那么整个过程是怎么样的呢?以我们自己的想法来走Vue的道路,其实也就是以Vue的原理为终点,我们来逆推一下实现过程。

本文代码皆为低配版本,很多地方都不严谨,比如 if(typeof obj === 'object')这是在判断obj是否为为一个对象,虽然obj也有可能是数组等其他类型的数据,但是本文为了简便,就直接这样写来表示判断对象,对于数组使用Array.isArray()。

改造数据

我们先来尝试写一个函数,用于改造对象:

为什么要先写这个函数呢? 因为改造数据是一个最基础也是最重要的步骤,之后所有的步骤都会依赖这一步。

// 代码 1.1
function defineReactive (obj,key,val) {
 Object.defineProperty(obj,key,{
  enumerable: true,
  configurable: true,
  get: function () {
   return val;
  },
  set: function (newVal) {
   //判断新值与旧值是否相等
   //判断的后半段是为了验证新值与旧值都为NaN的情况 NaN不等于自身
   if(newVal === val || (newVal !== newVal && value !== value)){
    return ;
   }
   val = newVal;
  }
 });
}

例如const obj = {},然后再调用defineReactive(obj,'a',2)方法,此时在函数内,val=2,然后每次获取obj.a的值的时候都是获取val的值,设置obj.a的时候也是设置val的值。(每次调用defineReactive都会产生一个闭包保存了val的值);

流程讨论

经过验证之后,发现这个函数确实可以使用的。然后我们来讨论一下响应的流程:

浅谈Vue 数据响应式原理

  1. 输入数据
  2. 改造数据(defineReactive())
  3. 如果数据变动 => 触发事件

我们来看第三步,数据变动如何触发之后的事件呢?仔细思考一下,如果要改变数据,那么必须先set数据,那么我们直接set()里面添加方法就ok了呀。

然后还有一个重要问题:

浅谈Vue 数据响应式原理

依赖收集

我们怎么知道数据改变之后要触发的是什么事件呢?在Vue中:

使用数据 => 视图; 使用了数据来渲染视图,那么在获取数据的时候收集依赖是最佳的时机,Vue在改造数据属性的时候生成一个Dep实例,用于收集依赖。

// 代码 1.2
class Dep {
 constructor(){
  //订阅的信息
  this.subs = [];
 }

 addSub(sub){
  this.subs.push(sub);
 }

 removeSub (sub) {
  remove(this.subs, sub);
 }

 //此方法的作用等同于 this.subs.push(Watcher);
 depend(){
  if (Dep.target) {
   Dep.target.addDep(this);
  }
 }
 //这个方法就是发布通知了 告诉你 有改变啦
 notify(){
  const subs = this.subs.slice()
  for (let i = 0, l = subs.length; i < l; i++) {
   subs[i].update();
  }
 }
}
Dep.target = null;

代码1.2就是Dep的部分代码,暂时只需要知道2个方法的作用就可以了

  1. depend() --- 可以理解为收集依赖的事件,不考虑其他方面的话 功能等同于addSub()
  2. notify() --- 这个方法更为直观了,执行所有依赖的update()方法。就是之后的改变视图啊 等等。

本篇主要讨论数据响应的过程,不深入讨论 Watcher类,所以Dep中的方法知道作用就可以了。

然后就是改变代码1.1了

//代码 1.3
function defineReactive (obj,key,val) {
 const dep = new Dep();

 Object.defineProperty(obj,key,{
  enumerable: true,
  configurable: true,
  get: function () {
   if(Dep.target){
    //收集依赖 等同于 dep.addSub(Dep.target)
    dep.depend()
   }
   return val;
  },
  set: function (newVal) {
   if(newVal === val || (newVal !== newVal && val !== val)){
    return ;
   }
   val = newVal;
   //发布改变
   dep.notify();
  }
 });
}

这代码中有一个疑点,Dep.target是什么?为什么要有Dep.target才会收集依赖呢?

  1. Dep是一个类,Dep.target是类的属性,并不是dep实例的属性。
  2. Dep类在全局可用,所以Dep.target在全局能访问到,可以任意改变它的值。
  3. get这个方法使用很平常,不可能每次使用获取数据值的时候都去调用dep.depend()。
  4. dep.depend()实际上就是dep.addSub(Dep.target)。
  5. 那么最好方法就是,在使用之前把Dep.target设置成某个对象,在订阅完成之后设置Dep.target = null。

验证

是时候来验证一波代码的可用性了

//代码 1.4

const obj = {};//这一句是不是感觉很熟悉 就相当于初始化vue的data ---- data:{obj:{}};

//低配的不能再低配的watcher对象(源码中是一个类,我这用一个对象代替了)
const watcher = {
 addDep:function (dep) {
  dep.addSub(this);
 },
 update:function(){
  html();
 }
}
//假装这个是渲染页面的
function html () {
 document.querySelector('body').innerHTML = obj.html;
}
defineReactive(obj,'html','how are you');//定义响应式的数据

Dep.target = watcher;
html();//第一次渲染界面
Dep.target = null;

此时浏览器上的界面是这样的

浅谈Vue 数据响应式原理

然后在下打开了控制台开始调试,输入:

obj.html = 'I am fine thank you'

然后就发现,按下回车的那一瞬间,奇迹发生了,页面变成了

浅谈Vue 数据响应式原理

结尾

Vue数据响应的设计模式和订阅发布模式有一点像,但是不同,每一个dep实例就是一个订阅中心,每一次发布都会把所有的订阅全部发布出去。

Vue的响应式原理其实还有很大一部分,本文主要讨论了Vue是如何让数据进行响应,但是实际上,一般的数据都是很多的,一个数据被多处使用,改变数据之后观察新值,如何观察、如何订阅、如何调度,都还有很大一部分没有讨论。主要的三个类Dep(收集依赖)、Observer(观察数据)、Watcher(订阅者,若数据有变化通知订阅者),都只提了一点点。

之前写有一篇Vue响应式----数组变异方法,针对Vue中对数组的改造进行讨论。当然之后有更多其他的文章,整个数据响应流程还有很多内容,三个主要的类都还没有讨论完。

其实阅读源码不仅仅是为了知道源码是如何工作的,更重要的是学习作者的思路与方法,我写的文章都不长,希望自己能够每次专注一个点,能够真真实实领悟到这一个点的原理。当然也想控制阅读时间,免得大家看到一半就关闭了。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
javascript call和apply方法
Nov 24 Javascript
jquery设置元素的readonly和disabled的写法
Sep 22 Javascript
JavaScript-RegExp对象只能使用一次问题解决方法
Jun 23 Javascript
高性能JavaScript 重排与重绘(2)
Aug 11 Javascript
详解JS中的立即执行函数
Feb 24 Javascript
vue技术分享之你可能不知道的7个秘密
Apr 09 Javascript
使用vue根据状态添加列表数据和删除列表数据的实例
Sep 29 Javascript
详解js静态检查工具eslint配置文件
Nov 23 Javascript
angular6的table组件开发的实现示例
Dec 26 Javascript
webpack4.0+vue2.0利用批处理生成前端单页或多页应用的方法
Jun 28 Javascript
使用p5.js临摹动态图片
Nov 04 Javascript
vue项目中在可编辑div光标位置插入内容的实现代码
Jan 07 Javascript
浅谈Vue响应式(数组变异方法)
May 07 #Javascript
在HTML文档中嵌入JavaScript的四种方法
May 07 #Javascript
详解JavaScript的BUG和错误
May 07 #Javascript
vue实现个人信息查看和密码修改功能
May 06 #Javascript
基于vue-element组件实现音乐播放器功能
May 06 #Javascript
VueJs组件之父子通讯的方式
May 06 #Javascript
vue自动化表单实例分析
May 06 #Javascript
You might like
apache+php完美解决301重定向的两种方法
2011/06/08 PHP
Laravel路由设定和子路由设定实例分析
2016/03/30 PHP
PHP中抽象类和抽象方法概念与用法分析
2016/05/24 PHP
php实现压缩合并js的方法【附demo源码下载】
2016/09/22 PHP
PHP给前端返回一个JSON对象的实例讲解
2018/05/31 PHP
ThinkPHP5+UEditor图片上传到阿里云对象存储OSS功能示例
2019/08/05 PHP
解决PhpStorm64不能启动的问题
2020/06/20 PHP
优化 JavaScript 代码的方法小结
2009/07/16 Javascript
jquery和ajax的关系详细介绍
2013/11/29 Javascript
js,jquery滚动/跳转页面到指定位置的实现思路
2014/06/03 Javascript
Javascript基础教程之比较操作符
2015/01/18 Javascript
jQuery团购倒计时特效实现方法
2015/05/07 Javascript
轻松掌握JavaScript中的Math object数学对象
2016/05/26 Javascript
jQuery UI Bootstrap是什么?
2016/06/17 Javascript
JavaScript中的对象和原型(一)
2016/08/12 Javascript
JS正则表达式之非捕获分组用法实例分析
2016/12/28 Javascript
将input框中输入内容显示在相应的div中【三种方法可选】
2017/05/08 Javascript
详解Angular2 之 结构型指令
2017/06/21 Javascript
angular中子控制器向父控制器传值的实例
2018/10/08 Javascript
Cordova(ionic)项目实现双击返回键退出应用
2019/09/17 Javascript
微信小程序学习总结(三)条件、模板、文件引用实例分析
2020/06/04 Javascript
Python常见数据结构详解
2014/07/24 Python
Python 遍历列表里面序号和值的方法(三种)
2017/02/17 Python
python进程间通信Queue工作过程详解
2019/11/01 Python
python中设置超时跳过,超时退出的方式
2019/12/13 Python
如何使用python-opencv批量生成带噪点噪线的数字验证码
2020/12/21 Python
名企HR怎样看待求职信
2014/02/23 职场文书
酒后驾车标语
2014/06/30 职场文书
领导班子四风表现材料
2014/08/23 职场文书
11.9消防日宣传标语
2014/10/08 职场文书
岗位职责范本大全
2015/02/26 职场文书
2015年推普周活动总结
2015/03/27 职场文书
售后服务质量承诺书
2015/04/29 职场文书
学生病假条范文
2015/08/17 职场文书
Netty分布式客户端处理接入事件handle源码解析
2022/03/25 Java/Android
SQL使用复合索引实现数据库查询的优化
2022/05/25 SQL Server