Vue3响应式原理解析

阿里云国内75折 回扣 微信号:monov8
阿里云国际,腾讯云国际,低至75折。AWS 93折 免费开户实名账号 代冲值 优惠多多 微信号:monov8 飞机:@monov6

前言

今年上半年开始自己开始在新项目中使用 Vue3 进行开发相比较于 Vue2 来说最大的变化就是 composition Api 代替了之前的 options Api更像是 React Hooks 函数式组件的编程方式。

Vue3相对于Vue2响应式原理也发生了变化由原先的 Object.defineproperty 改成了使用 Proxy 替代。Proxy 相对于 Object.defineproperty 有以下几个优化点

  • 对象新增属性不再需要手动 $set 添加响应式Proxy 默认会监听动态添加属性和属性的删除等操作。
  • 消除无法监听数组索引length 属性等等不再需要在数组原型对象上重写数组的方法。
  • Object.defineproperty 是劫持所有对象属性的 get/set 方法,需要遍历递归去实现Proxy 是代理整个对象。
  • Vue2 只能拦截对象属性的 getset 操作,而 Proxy 拥有 13 种拦截方法。

所有这些优化都指向了同一个点Vue3 将拥有更快的响应速度。下面将结合代码揭秘 Vue3 实现响应式的原理。

Proxy

Proxy 能够为另一个对象创建代理该代理可以拦截和重新定义该对象的基本操作,例如获取、设置和定义属性。

Proxy 接受两个参数

  • 要代理的原始对象。
  • 一个对象它定义了哪些操作将被拦截以及如何重新定义被拦截的操作。
const target = {name: "ts",age: 18
};
const handler = {};
const proxy = new Proxy(target, handler); 

我们可以在 handler 对象上定义函数做自定义代理:

const target = {name: "ts",age: "18"
};
const handler = {get(target, key, receiver) {console.log(`访问属性${key}值`)return Reflect.get(target, key, receiver)},set(target, key, value, receiver) {console.log(`设置属性${key}值`)return Reflect.set(target, key, value, receiver)},
};
const proxy = new Proxy(target, handler);
console.log(proxy.name)
proxy.name = 'jkl';
proxy.sex = 'male'; 

打印

注意

  • set 方法要求返回一个布尔值而 Reflect.set 方法刚好就是一个返回一个布尔值直接 return 就好了。
  • sex 属性是我们后面新增的但是也能在 getset 中拦截到说明 Proxy 是自动给新增属性添加响应式而不需要手动 $set 添加响应式。

通过对 Proxy 用法的基本介绍我们发现 ProxyObject.defineproperty 用法有一个相似之处它们内部都有 getset 方法我们可以在 getset 方法中拦截和重新定义一些逻辑处理和 Object.defineproperty 一样我们可以在 Proxyget 方法中进行依赖收集即 track 操作在 set 方法中进行触发更新即 trigger 操作。

Reflect

Reflect定义

Reflect 是一个内置的对象与 Math 类似它提供拦截 JavaScript 操作的方法这些方法与 Proxy handlers 提供的的方法是一一对应的且 Reflect 不是一个函数对象即不能进行实例化其所有属性和方法都是静态的。

Reflect.get/set 参数说明

  • target 指的是原始数据对象。
  • key 指的是操作的属性名。
  • newVal 指的是操作接收到的最新值。
  • receiver 指向的是当前操作正确的上下文代理对象。

receiver 作用

receiver 是为了在执行对应的拦截操作的方法时能传递正确的 this 上下文。

reactive

基于上面对 Proxy 的基本使用我们可以试着实现 reactive在 Vue3 中 reactive 是返回一个 Proxy 的方法接受一个对象作为参数

基本实现

export const reactive = (target: object) => {return new Proxy(target, {get(target, key, receiver) {return Reflect.get(target, key, receiver)},set(target, key, value, receiver) {return Reflect.set(target, key, value, receiver)}})
} 

如果 target 对象存在深层次结构我们就需要递归实现

递归完整实现

const isObject = target => target !== null && typeof target == 'object'
export const reactive = (target: object) => {return new Proxy(target, {get(target, key, receiver) {console.log(`访问属性${key}值`)const result = Reflect.get(target, key, receiver)// 判断result是否是引用类型是需要递归处理if (isObject(result)) {return reactive(result)}return result},set(target, key, value, receiver) {console.log(`设置属性${key}值`)return Reflect.set(target, key, value, receiver)}})
} 

测试

<!DOCTYPE html>
<html lang="en">

<head><meta charset="UTF-8"><meta http-equiv="X-UA-Compatible" content="IE=edge"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Document</title>
</head>

<body><script type="module"> import { reactive } from './reactive.js'const proxy = reactive({name: 'ts', age: 18})console.log(proxy.name);console.log(proxy.age);proxy.name = 'jkl';proxy.age = 20; </script>
</body>

</html> 

收集依赖和触发更新

收集依赖和触发更新是 Vue3 响应式最核心的部分。这里涉及到三个核心概念effecttracktrigger依赖收集依赖触发更新

访问代理对象 target 属性会触发 get 方法在这里会进行依赖收集即执行 track 方法。收集的依赖存储在 deps 里。修改 target 对象属性时触发 set 方法在这里会进行触发更新的操作即依次执行 deps 里面的依赖。

存储容器说明

  • 选择 weakMap 类型作为容器是因为 weakMap 对键的引用是弱类型当外部没有对键引用时weakMap会自动删除保证对象能被垃圾回收。
  • Map 类型对键的引用是强引用即便外部没有对该对象保持引用但至少还存在 Map 本身对该对象的引用关系因此会导致该对象不能及时的被垃圾回收。
  • 对应的响应式数据对象作为 targetMap 的键存储和当前响应式数据对象相关的依赖关系 depsMapdepsMap 存储的就是和当前响应式对象的每一个 key 对应的具体依赖。
  • deps 作为 depsMap 每个 key 对应的依赖集合因为每个响应式数据可能在多个副作用函数中被使用并且 Set 类型用于自动去重的能力。

effect

effect 依赖里面放着数据更新的逻辑通常我们放在一个函数里面。

// activeEffect 表示当前正在走的 effect
let activeEffect = null;
export const effect = (fn:Function) => {activeEffect = fnfn()activeEffect = null
} 

这里使用一个全局变量 activeEffect 来收集当前正在走的副作用函数并且初始化的时候调用一下。

let age = 18;
let result;
const effect = () => result = age * 2
age = 20;
effect();
console.log(result) // 40 

为了让大家理解 effect上面这段代码是一个比较形象的例子age 是一个变量effect 是副作用函数当 age 发生了变化 age = 20这时候我们调用 effect()更新了 result 值。在这里我们是手动写的调用 effect()在真实响应式流程中我们如何进行依赖收集以及自动触发更新 effect

track

track 函数用来进行依赖收集即把依赖于变量的 effect 函数收集起来放在 deps 里面deps 是一个 Set 数据结构。

const targetMap = new WeakMap()
export const track = (target, key) => {// 没有activeEffect就不进行追踪if (!activeEffect) return// 获取target的依赖图let depsMap = targetMap.get(target)// 没有就新建if (!depsMap) {depsMap = new Map()targetMap.set(target, depsMap)}// 获取key所对应依赖的集合let deps = depsMap.get(key)// 没有就新建if (!deps) {deps = new Set()depsMap.set(key, deps)}// 判断activeEffect是否存在不存在才添加防止重复添加if (!deps.has(activeEffect)) {deps.add(activeEffect)}
} 

在介绍 Proxy 的时候我们提到“我们会在 Proxy 的 get 方法中进行依赖收集即 track 操作”现在我们可以把 track 添加到 get 方法中了

const isObject = target => target !== null && typeof target == 'object'
export const reactive = (target: object) => {return new Proxy(target, {get(target, key, receiver) {console.log(`访问属性${key}值`)const result = Reflect.get(target, key, receiver)// 收集依赖track(target,key)// 判断result是否是引用类型是需要递归处理if (isObject(result)) {return reactive(result)}return result},set(target, key, value, receiver) {console.log(`设置属性${key}值`)return Reflect.set(target, key, value, receiver)}})
} 

trigger

实现 trigger

const targetMap = new WeakMap()
export const trigger = (target, key) => {// 获取target的依赖图const depsMap = targetMap.get(target)// 没有说明没有被追踪就returnif (!depsMap) return// 获取key所对应依赖的集合const deps = depsMap.get(key)// 遍历依赖的集合依次执行副作用函数if (deps) {deps.forEach(effect => effect())}
} 

在介绍 Proxy 的时候我们提到“在 set 方法中进行触发更新即 trigger 操作”现在我们可以把 trigger 添加到 set 方法中了

const isObject = target => target !== null && typeof target == 'object'
export const reactive = (target: object) => {return new Proxy(target, {get(target, key, receiver) {console.log('访问属性"+key+"值')const result = Reflect.get(target, key, receiver)// 收集依赖track(target,key)// 判断result是否是引用类型是需要递归处理if (isObject(result)) {return reactive(result)}return result},set(target, key, value, receiver) {console.log(`设置属性${key}值`)// 触发更新trigger(target, key)return Reflect.set(target, key, value, receiver)}})
} 

完整代码

core.js

// activeEffect 表示当前正在走的 effect
let activeEffect = null
export const effect = fn => {activeEffect = fnfn()activeEffect = null
}

const targetMap = new WeakMap()
export const track = (target, key) => {// 没有activeEffect就不进行追踪if (!activeEffect) return// 获取target的依赖图let depsMap = targetMap.get(target)// 没有就新建if (!depsMap) {depsMap = new Map()targetMap.set(target, depsMap)}// 获取key所对应依赖的集合let deps = depsMap.get(key)// 没有就新建if (!deps) {deps = new Set()depsMap.set(key, deps)}// 判断activeEffect是否存在不存在才添加防止重复添加if (!deps.has(activeEffect)) {deps.add(activeEffect)}
}

export const trigger = (target, key) => {// 获取target的依赖图const depsMap = targetMap.get(target)// 没有说明没有被追踪就returnif (!depsMap) return// 获取key所对应依赖的集合const deps = depsMap.get(key)console.log(deps, 'deps=====')// 遍历依赖的集合依次执行副作用函数if (deps) {deps.forEach(effect => effect())}
} 

reactive.js

import { track, trigger } from './core.js'
const isObject = target => target !== null && typeof target == 'object'
export const reactive = target => {return new Proxy(target, {get(target, key, receiver) {console.log(`访问属性${key}值`)const result = Reflect.get(target, key, receiver)// 收集依赖track(target, key)// 判断result是否是引用类型是需要递归处理if (isObject(result)) {return reactive(result)}return result},set(target, key, value, receiver) {console.log(`设置属性${key}值`)// 触发更新trigger(target, key)return Reflect.set(target, key, value, receiver)}})
} 

index.html

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><meta http-equiv="X-UA-Compatible" content="IE=edge"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Document</title>
</head>
<body><div class="box"></div><button>修改</button><script type="module"> import { reactive } from './reactive.js'import { effect } from './core.js'const user = reactive({name: 'ts'})effect(() => { document.querySelector('.box').innerText = `${user.name}` })document.querySelector('button').onclick = function () {user.name = 'jkl'} </script>
</body>
</html> 

效果

总结

Vue2 和 Vue3 实现响应式的思路或者核心都是相同的即数据劫持/对象代理自定义get / set、依赖收集、触发更新。Vue3 使用 Proxy 实现响应式是对 Object.defineproperty 实现方案存在缺陷的一种优化。

最后

为大家准备了一个前端资料包。包含54本2.57G的前端相关电子书《前端面试宝典附答案和解析》难点、重点知识视频教程全套。



有需要的小伙伴可以点击下方卡片领取无偿分享

阿里云国内75折 回扣 微信号:monov8
阿里云国际,腾讯云国际,低至75折。AWS 93折 免费开户实名账号 代冲值 优惠多多 微信号:monov8 飞机:@monov6
标签: vue