当前位置: 首页 > news >正文

建立网站站建设可以吗谷歌浏览器网页版在线

建立网站站建设可以吗,谷歌浏览器网页版在线,外贸做企业什么网站,手机软件开发网站文章目录【Vue3源码】第一章 effect和reactive前言1、实现effect函数2、封装track函数(依赖收集)3、封装reactive函数4、封装trigger函数(依赖触发)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单元测试环境,可以查看我的上一篇文章

  1. 首先我们新建一个effect.ts文件并且封装一个effect函数
//用来暂存effect传递的参数fn
let activeEffect;export const effect = (fn) => {activeEffect = fnfn()
};

​ 为什么我们还要在函数定义一个activeEffect块级作用域变量呢?

​ 因为activeEffect要帮我们暂存用户传进来的fn参数,fn的类型是一个函数,那么fn暂存到了activeEffect变量后为了不让下次传递的参数覆盖掉了我们的之前的fn,Vue就该把它存到一个“仓库”里保存着,方便后续管理,这个过程叫依赖收集

​ 我们再优化一下代码方便后续管理。

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代理对象实现了数据响应式,Proxy(MDN对Proxy的介绍)代理多达13种捕获器它们可以完美的监听任何方式的数据改变,完美的解决了的vue2使用Object.defineProperty监听不到数组下标缺点。

不过说到缺点:真是非常的尴尬。。。不管Vue2的劫持还是Vue3的代理对有深层嵌套的对象还是要用递归去处理并且返回多层的响应式对象。

我们今天的文章只处理没有嵌套关系的对象,不深入多层嵌套对象的递归处理,下次再处理这个逻辑。

极简版reactive函数很简单,return反射的结果之前进行依赖收集即可,传入的target正是代理对象,key是正在使用对象的key。

新建一个reactive文件

import { track, trigger } from "./effect"export 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)});
});
http://www.yayakq.cn/news/550050/

相关文章:

  • 网站公司如何推广网站合肥新房在售楼盘
  • wordpress标签云404错误哈尔滨seo推广公司
  • 男女直接做的视频网站免费观看免费自助建站哪个网站最好
  • 怎么做淘宝客个人网站深圳网站建设简介
  • 怎样用代码做网站长春建站模板
  • 青岛网站快速备案wordpress redis
  • 给公司做网站要花多钱广告效果图用什么软件做
  • 改变网站的域名空间网页设计高清素材
  • 营销型网站建设熊掌号台州市知名专业做网站
  • 郑州数码网站建设服务公司重庆工程建筑信息网
  • 做特卖的网站上品折扣张家港手机网站制作
  • 网站更新和维护怎么做如何让本机做网站让内网访问
  • 帮人做彩票网站沧州1 1 网站建设
  • 请别人做网站的缺点免费dns二级域名
  • 软件开发工程师薪资水平北京债务优化公司
  • 重庆品牌网站建设怎么样百度电脑怎么用wordpress
  • 公司做网站推广有没有用网站开发怎么自动获取位置
  • 江苏品牌网站建设电话淘宝热搜关键词排行榜
  • 枞阳网站建设wordpress pdf阅读器
  • 南宁老牌网站建设公司网站404页面查询
  • 织梦手机网站教程视频教程昆明hph网站建设
  • 石家庄住房和城乡建设局官方网站现在淘客做网站还行吗
  • 难道做网站的工资都不高吗山东省建设厅招标网站首页
  • 深圳自定义网站开发软件开发培训机构前十
  • 嘉兴行业网站建设徐州做网站的公司有哪些
  • 手机网站的页面大小网站要备案吗
  • 郑州网站制作推广个人网站主题
  • 集团网站品牌建设特点要怎样建设网站
  • 最适合seo的网站源码机关事业单位网站建设
  • 西安西郊网站建设seo就业前景怎么样