likes
comments
collection
share

(二)手写 Vue 源码 —— 渲染更新

作者站长头像
站长
· 阅读数 60
上一篇我们实现了数据到视图的映射,这一篇我们实现数据驱动,当数据改变的时候,我们无需手动操作 DOM ,视图会自动更新。

模拟更新

现在当我们改变数据的时候,data 数据源已经改变了,但是视图并没有改变,需要手动调用 vm._update(vm._render()) 才能更新视图,所以我们需要一个机制在数据变动的时候自动去更新。

(二)手写 Vue 源码 —— 渲染更新

一、定义 Watcher

src/observer/watcher.js

// 全局变量 id 每次 new Watcher 都会自增
let id = 0;

class Watcher {
	constructor(vm,exprOrFn, cb, options) {
		this.vm = vm;
		this.exprOrFn = exprOrFn;
		this.cb = cb; //回调函数 比如在watcher更新之前可以执行beforeUpdate方法
		this.options = options; //额外的选项 true代表渲染watcher
		this.id = id++;
		if (typeof exprOrFn === 'function') {
			this.getter = exprOrFn;
		}
		this.get();
	}

	get () {
		this.getter();
	}
}

export default Watcher;

在 observer 文件夹下新建 watcher.js 代表和观察者相关 这里首先介绍 Vue 里面使用到的观察者模式,我们可以把 Watcher 当做观察者 它需要订阅数据的变动 当数据变动之后 通知它去执行某些方法 其实本质就是一个构造函数 初始化的时候会去执行 get 方法

二、创建渲染 Watcher

src/lifecycle.js

(二)手写 Vue 源码 —— 渲染更新

在组件挂载方法里面定义一个渲染 Watcher 主要功能就是执行核心渲染页面的方法

三、定义 Dep

src/observer/dep.js

(二)手写 Vue 源码 —— 渲染更新

Dep 是一个构造函数,可以把他理解为观察者模式里面的被观察者在 subs 里面收集 watcher ;当数据变动的时候通知自身 subs 所有的 watcher 更新;

四、对象的依赖收集

Object.defineProperty数据劫持核心 兼容性在ie9以及以上

src/observer/index.js

(二)手写 Vue 源码 —— 渲染更新 上诉代码就是依赖收集和派发更新的核心 其实就是在数据被访问的时候 把我们定义好的渲染 Watcher 放到 dep 的 subs 数组里面 同时把 dep 实例对象也放到渲染 Watcher 里面去 数据更新时就可以通知 dep 的 subs 存储的 watcher 更新

五、 完善watcher

src/observer/watcher.js

(二)手写 Vue 源码 —— 渲染更新

(二)手写 Vue 源码 —— 渲染更新

watcher 在调用 getter 方法前后分别把自身赋值给 Dep.target 方便进行依赖收集;

六、完善 dep

src/observer/dep.js

(二)手写 Vue 源码 —— 渲染更新

(二)手写 Vue 源码 —— 渲染更新

定义相关的方法把收集依赖的同时把自身也放到 watcher 的 deps 容器里面去

这时对象的更新已经可以满足了 但是如果是数组 类似{a:[1,2,3]} a.push(4) 并不会触发自动更新 因为我们数组并没有收集依赖

七、数组的依赖收集

  • childOb 就是Observer实例,表示 属性的值依然是一个对象 包含数组和对象 childOb指代的就是Observer实例对象 里面的dep进行依赖收集
  • 比如{a:[1,2,3]} 属性a对应的值是一个数组 观测数组的返回值就是对应数组的Observer实例对象
  • e.__ob__代表e已经被响应式观测了 但是没有收集依赖 所以把他们收集到自己的Observer实例的dep里面

(二)手写 Vue 源码 —— 渲染更新

(二)手写 Vue 源码 —— 渲染更新

(二)手写 Vue 源码 —— 渲染更新

如果对象属性的值是一个数组,那么执行 childOb.dep.depend() 收集数组的依赖;如果数组里面还包含数组,需要递归遍历收集,因为只有访问数据触发了 get 才会去收集依赖,一开始只是递归对数据进行响应式处理无法收集依赖,这两点要分清。

八、数组的派发更新

src/observer/array.js

(二)手写 Vue 源码 —— 渲染更新

ob 指的是数组对应的 observer 实例,我们在 get 的时候判断如果属性的值还是对象name就在 observer 实例的 dep 收集依赖,所以这里是一一对应的,可以直接更新。

九、总结

从设计模式理解 Vue 响应式原理

数据劫持-->模板解析-->模板渲染-->数据变化视图自动更新

(二)手写 Vue 源码 —— 渲染更新

文章参考:

作者: 前端鲨鱼哥

转载自:https://juejin.cn/post/7287389831193346106
评论
请登录