上海seo网站建设,上传网站再备案,做网站需要了解,甘肃省建设厅安全员官方网站文章目录【Vue3源码】第一章 effect和reactive前言1、实现effect函数2、封装track函数#xff08;依赖收集#xff09;3、封装reactive函数4、封装trigger函数#xff08;依赖触发#xff09;5、单元测试【Vue3源码】第一章 effect和reactive
前言
今天就正式开始Vue3源码…
文章目录【Vue3源码】第一章 effect和reactive前言1、实现effect函数2、封装track函数依赖收集3、封装reactive函数4、封装trigger函数依赖触发5、单元测试【Vue3源码】第一章 effect和reactive
前言
今天就正式开始Vue3源码学习了那么很多初学者包括我在看vue源码时都会非常迷茫不知从何下手所以我们在学习源码时应该反其道而行之剔除掉源码中包括Tree-Shaking、TypeScript类型约束、特性开关、错误处理等操作只了解其核心原理大大提高学习效率减少学习成本 如果你还不了解Vue源码设计“Tree-Shaking、TypeScript类型约束、特性开关、错误处理” 可以购买一本《Vue.js设计与实现》在第 2 章 “框架设计的核心要素” 有详细介绍了这些操作并且有举例子解释了为什么要这么设计。 好了正式开始学习想要了解vue3的源码我们可以先从reactivity文件夹入手。
那么reactivity文件夹是什么呢 Reactivity里面写的是vue最最重要的功能那就是“vue被人津津乐道并且顶顶大名的响应式源码“。
今天我们就简单的从零开始实现一遍effect函数影响效果reactive函数代理或者劫持。
在reactive函数中还包括两个功能依赖收集和依赖触发他们分别是track函数依赖收集和trigger函数依赖触发在今天的文章中我都会一一实现一遍。
1、实现effect函数
effect可以说是响应式系统的核心所以我们学习vue3源码时推荐从effect入手。 如果你还没有搭建好jest单元测试环境可以查看我的上一篇文章 首先我们新建一个effect.ts文件并且封装一个effect函数
//用来暂存effect传递的参数fn
let activeEffect;export const effect (fn) {activeEffect fnfn()
}; 为什么我们还要在函数定义一个activeEffect块级作用域变量呢
因为activeEffect要帮我们暂存用户传进来的fn参数fn的类型是一个函数那么fn暂存到了activeEffect变量后为了不让下次传递的参数覆盖掉了我们的之前的fnVue就该把它存到一个“仓库”里保存着方便后续管理这个过程叫依赖收集。
我们再优化一下代码方便后续管理。
class ReactiveEffect {private _fn;constructor(fn) {this._fn fn;}run() {activeEffect this;this._fn();}
}export const effect (fn) {const _effect new ReactiveEffect(fn);_effect.run();}; 那么怎么才能做到依赖收集呢往下看⬇️
2、封装track函数依赖收集
在effect.ts文件中封装track函数
为了方便理解我把targetMap抽象的比做一个仓库方便大家理解作为一个仓库自然就得有仓库的管理规则货物进来总不能不分类就乱七八糟直接存进去吧
所以我们就根据传递进来的参数target一个对象key该对象中的key作为仓库的类别一一细化分类这个仓库里包含的货物。
//targetMap就是一个收集effect传递过来的参数fn的“总仓库”
const targetMap new WeakMap();export function track(target, key) {//根据target一级分类let depsMap targetMap.get(target);//如果这个货物是第一次存我们就新建这个分类if (!depsMap) {depsMap new Map();targetMap.set(target, depsMap);}//根据key二级分类let dep depsMap.get(key);//如果这个货物是第一次存我们就新建这个二级分类if (!dep) {dep new Set();depsMap.set(key, dep);}//最后把收集到的activeEffect存入这个细化的分类里dep.add(activeEffect);
} 所以targetMap作为一个总仓库它通过传入的target和key进行分类保存我们的activeEffect。
我画一个drawio图看一下就可以明白里面的逻辑。 那么我们的track函数接收的target和key参数又是怎么来的往下看⬇️
3、封装reactive函数
我们都知道vue3使用的是Proxy代理对象实现了数据响应式ProxyMDN对Proxy的介绍代理多达13种捕获器它们可以完美的监听到任何方式的数据改变完美的解决了的vue2使用Object.defineProperty时监听不到数组下标缺点。 不过说到缺点真是非常的尴尬。。。不管Vue2的劫持还是Vue3的代理对有深层嵌套的对象还是要用递归去处理并且返回多层的响应式对象。 我们今天的文章只处理没有嵌套关系的对象不深入多层嵌套对象的递归处理下次再处理这个逻辑。 极简版reactive函数很简单return反射的结果之前进行依赖收集即可传入的target正是代理对象key是正在使用对象的key。
新建一个reactive文件
import { track, trigger } from ./effectexport const reactive (raw) {return new Proxy(raw,{get(target,key,receiver) {const res Reflect.get(target,key,receiver)//do something 收集依赖track(target,key)return res},set(target,key,value,receiver) {const res Reflect.set(target,key,value,receiver)// do something 触发依赖trigger(target,key)return res},})
}4、封装trigger函数依赖触发
上一步封装好了reavtive函数的还差最后一个trigger函数我们还没有封装。
其实依赖触发也很简单通过传入的target和key货物的分类类别我们就可以从总仓库里取出收集到的对应依赖并且再出触发它
在effect.ts文件中封装trigger
export function trigger(target, key) {let depsMap targetMap.get(target);let dep depsMap.get(key);for (let effect of dep) {effect.run();}
}5、单元测试
我们新建在test文件夹下新建一个 effect.spec.ts文件
然后就可以打上断点跟着断点走一遍Vue响应式的执行帮助我们理解Vue的逻辑。
我测试的代码如下
import { effect } from ../effect;
import { reactive } from ../reactive;describe(effect, () {it(happy path, () {const user reactive({age: 10,name:www,newObj:{objAge:11}});let nextAge;let age2effect(() {nextAge user.age 1;});//无法代理深层嵌套的对象effect(() {age2 user.newObj.objAge;});expect(nextAge).toBe(11);user.age;expect(nextAge).toBe(12);user.age 99expect(nextAge).toBe(100)expect(age2).toBe(11)//对于深层嵌套的对象由于没有封装递归的逻辑所以监听不到user.newObj.objAge //理论上来说age2应该跟着user.newobj.objeAge响应式变成12而结果却没有变化expect(age2).toBe(11)});
});