Vue的响应式系统

Vue 最独特的特性之一,是其非侵入性的响应式系统。数据模型仅仅是普通的JavaScript 对象,而当你修改它们时,视图会进行更新,这使得状态管理非常简单直接,我们可以只关注数据本身,而不用手动处理数据到视图的渲染,避免了繁琐的 DOM 操作,提高了开发效率。

vue 的响应式系统依赖于三个重要的类:Dep 类、Watcher 类、Observer 类,然后使用发布订阅模式的思想将他们揉合在一起(不了解发布订阅模式的可以看我之前的文章发布订阅模式与观察者模式)。

Observer

Observe扮演的角色是发布者,他的主要作用是调用defineReactive函数,在defineReactive函数中使用Object.defineProperty 方法对对象的每一个子属性进行数据劫持/监听。

部分代码展示

defineReactive函数,Observe的核心,劫持数据,在setter中向Dep(调度中心)添加观察者,在getter中通知观察者更新。

function defineReactive(obj, key, val, customSetter, shallow){//监听属性key//关键点:在闭包中声明一个Dep实例,用于保存watcher实例var dep = new Dep();var getter = property && property.get;var setter = property && property.set;if(!getter && arguments.length === 2) {val = obj[key];}//执行observe,监听属性key所代表的值val的子属性var childOb = observe(val);Object.defineProperty(obj, key, {enumerable: true,configurable: true,get: function reactiveGetter() {//获取值var value = getter ? getter.call(obj) : val;//依赖收集:如果当前有活动的Dep.target(观察者--watcher实例)if(Dep.target) {//将dep放进当前观察者的deps中,同时,将该观察者放入dep中,等待变更通知dep.depend();if(childOb) {//为子属性进行依赖收集//其实就是将同一个watcher观察者实例放进了两个dep中//一个是正在本身闭包中的dep,另一个是子属性的depchildOb.dep.depend();}}return value},set: function reactiveSetter(newVal) {//获取valuevar value = getter ? getter.call(obj) : val;if(newVal === value || (newVal !== newVal && value !== value)) {return}if(setter) {setter.call(obj, newVal);} else {val = newVal;}//新的值需要重新进行observe,保证数据响应式childOb = observe(newVal);//关键点:遍历dep.subs,通知所有的观察者dep.notify();}});
}

Dep

Dep 扮演的角色是调度中心/订阅器,主要的作用就是收集观察者Watcher和通知观察者目标更新。每个属性拥有自己的消息订阅器dep,用于存放所有订阅了该属性的观察者对象,当数据发生改变时,会遍历观察者列表(dep.subs),通知所有的watch,让订阅者执行自己的update逻辑。

部分代码展示

Dep的设计比较简单,就是收集依赖,通知观察者

//Dep构造函数
var Dep = function Dep() {this.id = uid++;this.subs = [];
};
//向dep的观察者列表subs添加观察者
Dep.prototype.addSub = function addSub(sub) {this.subs.push(sub);
};
//从dep的观察者列表subs移除观察者
Dep.prototype.removeSub = function removeSub(sub) {remove(this.subs, sub);
};
Dep.prototype.depend = function depend() {//依赖收集:如果当前有观察者,将该dep放进当前观察者的deps中//同时,将当前观察者放入观察者列表subs中if(Dep.target) {Dep.target.addDep(this);}
};
Dep.prototype.notify = function notify() {// 循环处理,运行每个观察者的update接口var subs = this.subs.slice();for(var i = 0, l = subs.length; i < l; i++) {subs[i].update();}
};//Dep.target是观察者,这是全局唯一的,因为在任何时候只有一个观察者被处理。
Dep.target = null;
//待处理的观察者队列
var targetStack = [];function pushTarget(_target) {//如果当前有正在处理的观察者,将他压入待处理队列if(Dep.target) {targetStack.push(Dep.target);}//将Dep.target指向需要处理的观察者Dep.target = _target;
}function popTarget() {//将Dep.target指向栈顶的观察者,并将他移除队列Dep.target = targetStack.pop();
}

Watcher

Watcher扮演的角色是订阅者/观察者,他的主要作用是为观察属性提供回调函数以及收集依赖(如计算属性computed,vue会把该属性所依赖数据的dep添加到自身的deps中),当被观察的值发生变化时,会接收到来自dep的通知,从而触发回调函数。,

部分代码展示

Watcher类的实现比较复杂,因为他的实例分为渲染 watcher(render-watcher)、计算属性 watcher(computed-watcher)、侦听器 watcher(normal-watcher)三种,
这三个实例分别是在三个函数中构建的:mountComponent 、initComputed和Vue.prototype.$watch。

normal-watcher: 我们在组件钩子函数watch 中定义的,都属于这种类型,即只要监听的属性改变了,都会触发定义好的回调函数,这类watch的expression是我们写的回调函数的字符串形式。

computed-watcher: 我们在组件钩子函数computed中定义的,都属于这种类型,每一个 computed 属性,最后都会生成一个对应的 watcher 对象,但是这类 watcher 有个特点:当计算属性依赖于其他数据时,属性并不会立即重新计算,只有之后其他地方需要读取属性的时候,它才会真正计算,即具备 lazy(懒计算)特性。这类watch的expression是计算属性中的属性名。

render-watcher: 每一个组件都会有一个 render-watcher, 当 data/computed 中的属性改变的时候,会调用该 render-watcher 来更新组件的视图。这类watch的expression是function () {vm._update(vm._render(), hydrating);}

除了功能上的区别,这三种 watcher 也有固定的执行顺序,分别是:computed-render -> normal-watcher -> render-watcher

这样安排是有原因的,这样就能尽可能的保证,在更新组件视图的时候,computed 属性已经是最新值了,如果 render-watcher 排在 computed-render 前面,就会导致页面更新的时候 computed 值为旧数据。

这里我们只看其中一部分代码

function Watcher(vm, expOrFn, cb, options, isRenderWatcher) {this.vm = vm;if(isRenderWatcher) {vm._watcher = this;}vm._watchers.push(this);// optionsif(options) {this.deep = !!options.deep; //是否启用深度监听this.user = !!options.user; //主要用于错误处理,侦听器 watcher的 user为true,其他基本为falsethis.lazy = !!options.lazy; //惰性求职,当属于计算属性watcher时为truethis.sync = !!options.sync; //标记为同步计算,三大类型暂无} else {this.deep = this.user = this.lazy = this.sync = false;}//初始化各种属性和option//观察者的回调//除了侦听器 watcher外,其他大多为空函数this.cb = cb;this.id = ++uid$1; // uid for batchingthis.active = true;this.dirty = this.lazy; // for lazy watchersthis.deps = [];this.newDeps = [];this.depIds = new _Set();this.newDepIds = new _Set();this.expression = expOrFn.toString();// 解析expOrFn,赋值给this.getter// 当是渲染watcher时,expOrFn是updateComponent,即重新渲染执行render(_update)// 当是计算watcher时,expOrFn是计算属性的计算方法// 当是侦听器watcher时,expOrFn是watch属性的名字,this.cb就是watch的handler属性//对于渲染watcher和计算watcher来说,expOrFn的值是一个函数,可以直接设置getter//对于侦听器watcher来说,expOrFn是watch属性的名字,会使用parsePath函数解析路径,获取组件上该属性的值(运行getter)//依赖(订阅目标)更新,执行update,会进行取值操作,运行watcher.getter,也就是expOrFn函数if(typeof expOrFn === 'function') {this.getter = expOrFn;} else {this.getter = parsePath(expOrFn);}this.value = this.lazy ? undefined : this.get();
};
//取值操作
Watcher.prototype.get = function get() {//Dep.target设置为该观察者pushTarget(this);var vm = this.vm;//取值var value = this.getter.call(vm, vm);//移除该观察者popTarget();return value
};
Watcher.prototype.addDep = function addDep(dep) {var id = dep.id;if(!this.newDepIds.has(id)) {//为观察者的deps添加依赖depthis.newDepIds.add(id);this.newDeps.push(dep);if(!this.depIds.has(id)) {//为dep添加该观察者dep.addSub(this);}}
};
//当一个依赖改变的时候,通知它update
Watcher.prototype.update = function update() {//三种watcher,只有计算属性 watcher的lazy设置了true,表示启用惰性求值if(this.lazy) {this.dirty = true;} else if(this.sync) {//标记为同步计算的直接运行run,三大类型暂无,所以基本会走下面的queueWatcherthis.run();} else {//将watcher推入观察者队列中,下一个tick时调用。//也就是数据变化不是立即就去更新的,而是异步批量去更新的queueWatcher(this);}
};//update执行后,运行回调cb
Watcher.prototype.run = function run() {if(this.active) {var value = this.get();if(value !== this.value ||isObject(value) ||this.deep) {var oldValue = this.value;this.value = value;//运行 cb 函数,这个函数就是之前传入的watch中的handler回调函数if(this.user) {try {this.cb.call(this.vm, value, oldValue);} catch(e) {handleError(e, this.vm, ("callback for watcher \"" + (this.expression) + "\""));}} else {this.cb.call(this.vm, value, oldValue);}}}
};//对于计算属性,当取值计算属性时,发现计算属性的watcher的dirty是true
//说明数据不是最新的了,需要重新计算,这里就是重新计算计算属性的值。
Watcher.prototype.evaluate = function evaluate() {this.value = this.get();this.dirty = false;
};//收集依赖
Watcher.prototype.depend = function depend() {var this$1 = this;var i = this.deps.length;while(i--) {this$1.deps[i].depend();}
};

总结

Observe是对数据进行监听,Dep是一个订阅器,每一个被监听的数据都有一个Dep实例,Dep实例里面存放了N多个订阅者(观察者)对象watcher。

被监听的数据进行取值操作时(getter),如果存在Dep.target(某一个观察者),则说明这个观察者是依赖该数据的(如计算属性中,计算某一属性会用到其他已经被监听的数据,就说该属性依赖于其他属性,会对其他属性进行取值),就会把这个观察者添加到该数据的订阅器subs里面,留待后面数据变更时通知(会先通过观察者id判断订阅器中是否已经存在该观察者),同时该观察者也会把该数据的订阅器dep添加到自身deps中,方便其他地方使用。

被监听的数据进行赋值操作时(setter)时,就会触发dep.notify(),循环该数据订阅器中的观察者,进行更新操作。

vue响应式系统--observe、watcher、dep相关推荐

  1. data的值 如何初始化vue_理解Vue响应式系统

    深入理解 Vue 响应式系统 理解 Vue 响应式原理,到 computed.vuex 原理 前言 众所周知,一说到 vue 的响应式系统,就能马上想到 Object.defineProperty.数 ...

  2. Vue 响应式系统(二)- observe 工厂函数

    接上篇文章回到 initData 函数的最后一句代码: // observe data observe(data, true /* asRootData */) 调用了 observe 函数观测数据, ...

  3. 深入剖析Vue源码 - 响应式系统构建(上)

    从这一小节开始,正式进入Vue源码的核心,也是难点之一,响应式系统的构建.这一节将作为分析响应式构建过程源码的入门,主要分为两大块,第一块是针对响应式数据props,methods,data,comp ...

  4. vue源码分析-响应式系统(二)

    为了深入介绍响应式系统的内部实现原理,我们花了一整节的篇幅介绍了数据(包括data, computed,props)如何初始化成为响应式对象的过程.有了响应式数据对象的知识,上一节的后半部分我们还在保 ...

  5. 【Vue 3.0 新特性(四)】Vue 3.0 响应式系统原理

    文章前言 笔记来源:拉勾教育 大前端高薪训练营 阅读建议:内容较多,建议通过左侧导航栏进行阅读 Vue 3.0 响应式系统原理 基础回顾 Vue.js 响应式回顾 Proxy 对象实现属性监听 多层属 ...

  6. 一张图理清 Vue 3.0 的响应式系统

    点击上方 "程序员小乐"关注, 星标或置顶一起成长 每天凌晨00点00分, 第一时间与你相约 每日英文 Success is actually simple – when you ...

  7. vue 数组删除 dome没更新_详解Vue响应式原理

    摘要: 搞懂Vue响应式原理! 作者:浪里行舟 原文:深入浅出Vue响应式原理 Fundebug经授权转载,版权归原作者所有. 前言 Vue 最独特的特性之一,是其非侵入性的响应式系统.数据模型仅仅是 ...

  8. VUE 响应式原理源码:带你一步精通 VUE | 原力计划

    作者 | 爱编程的小和尚 责编 | 王晓曼 出品 | CSDN博客 学过 VUE 如果不了解响应式的原理,怎么能说自己熟练使用 VUE,要是没有写过一个简易版的 VUE 怎么能说自己精通 VUE,这篇 ...

  9. 深入浅出 Vue 响应式原理!

    作者 | 浪里行舟 责编 | 胡巍巍 Vue 最独特的特性之一,是其非侵入性的响应式系统.数据模型仅仅是普通的 JavaScript 对象.而当你修改它们时,视图会进行更新.这使得状态管理非常简单直接 ...

最新文章

  1. Springboot、Mybatis 事务示例
  2. 半导体理论(第2部分)半导体掺杂
  3. 微软职位内部推荐-SENIOR DEVELOPMENT LEAD
  4. Middleware课程01-概述
  5. 深入理解多线程(三)—— Java的对象头
  6. 三维空间几何变换原理[平移、旋转、错切]
  7. gdb core调试
  8. C/C++——从ctime使用到随便测一样冒泡排序和堆排序的效率
  9. JSP表单提交中文乱码解决方案
  10. vue后台如何刷新过期的token_Vue刷新token,判断token是否过期
  11. node使用ffmpeg拼接音频
  12. 倾斜摄影模型(.osgb)中心点位置的确定方法(SuperMap idesktop)
  13. AfterEffects 不支持 MKV 格式的解决办法
  14. 怎么批量修改pdf文件名
  15. win 7系统微信如何用代理服务器,win7系统电脑上使用微信的操作方法
  16. 推荐系统遇上深度学习(十二)--推荐系统中的EE问题及基本Bandit算法
  17. Tesla M40 下Ubuntu anaconda pycharm pytorch安装
  18. 2018年什么编程语言最值得学习
  19. 实验3 手写字体识别【机器学习】
  20. Ubuntu 安装PIXMA IP1180打印机

热门文章

  1. HDU 2448 Mining Station on the Sea
  2. android调用js函数方法,Android和JavaScript相互调用的方法
  3. AgileEAS.NET SOA 中间件平台5.2版本下载、配置学习(一):下载平台并基于直连环境运行...
  4. new OP 上线第二天总结:从哪里跌倒,就从哪里爬起来!
  5. 矢量图标库如何引入html,阿里巴巴矢量图标库 iconfont 的使用方法
  6. 19. 二元连续型随机变量,联合概率密度
  7. mysql数据库字符集_超详细的MySQL数据库字符集总结,值得收藏
  8. php strftime乱码,php – strftime错误的日期
  9. python快速倒包_Python 基础之import导包
  10. USACO 2004 Mar 赞助学费 finance 堆