3.1 一个稍微完善的Vue.js响应式系统

news2025/1/10 16:28:51

前文提要:3.0 响应式系统的设计与实现

1、设置一个合理的effect副作用函数

如上文所说,如果我们直接将简单的effect函数作为副作用函数,如果一个副作用函数不叫effect岂不是找不到了。解决方案也很简单,我们设定一个全局变量用于注册副作用函数,代码如下:

let activeEffect = null
function effect(fn) {
  // 当调用effect函数注册副作用函数时,将副作用函数fn赋值给activeEffect
  activeEffect = fn
  fn()
}

此时effect函数只是用于注册的功能,它接收一个参数fn,这个参数就是要注册的副作用函数,可以如下的方式使用effect函数:

effect(
  () => {document.body.innerText = obj.text}
)

此时将一个匿名函数作为副作用函数传递给effect,这个匿名函数将被赋值给activeEffect,这样只需要每次将activeEffect放入桶中就行了。如:

const bucket = new Set()
const obj = new Proxy(data, {
  get(target, key) {
    // 如果对象存在副作用函数,则放入桶中
    if(activeEffect) bucket.set(target, activeEffect)
    return target[key]
  },
  set(target, key, newVal) {
    // 每次赋值取出对象的副作用函数执行
    bucket.forEach(fn => fn())
    target[key] = newVal
    return true
  }
})

上述代码中的执行逻辑可以很清晰的看出,由于副作用函数存储到了activeEffect中,在get的时候可以直接将activeEffect函数放入桶中,这样就不需要依赖副作用函数叫什么名字了。

2、更加细化地绑定副作用函数

我们上面说所的所有的副作用函数其实都是直接绑定在对象上的,因为我们只给对象设定了桶,所以每次改变对象内无论哪个值都会将桶内的副作用函数全部执行。

这会导致什么问题呢,最明显的就是性能浪费,如果一个很大的对象,里面的每个元素都有很多副作用函数,那么我们在改变一个无关元素时也会执行全部副作用函数。

此时我们可以改变下桶的方式,使用一个WeakMap作为桶,WeakMap中的每一个键为一个响应式对象,它的值为一个Map,此时这个Map中的键为对象内的属性名,这个键的值为Set,此时Set内部装的是对象元素的副作用函数。如图所示:
在这里插入图片描述

代码实现如下:

// WeakMap的说明见下文
const bucket = new WeakMap()
  const obj = new Proxy(data, {
    get(target, key) {
      // 没有副作用函数直接返回
      if(!activeEffect) return target[key]
      // 获取当前对象内的所有元素的Map
      let depsMap = bucket.get(target)
      // 当前对象还没有Map时新建一个
      if(!depsMap) {
        bucket.set(target, (depsMap = new Map()))
      }
      // 获取当前元素的所有副作用函数的Set
      let deps = depsMap.get(key)
      // 没有时新建并添加副作用函数
      if(!deps) depsMap.set(key, (deps = new Set()))
      deps.add(activeEffect)
      return target[key]
    },
    set(target, key, newVal) {
      target[key] = newVal
      const depsMap = bucket.get(target)
      if(!depsMap) return
      const deps = depsMap.get(key)
      deps && deps.forEach(fn => fn())
      return true
    }
  })
}

其实代码看起来很复杂,其实只有三个容器,WeakMap,Map,Set,这里WeakMap是用于将所有的对象作为键值(key)放入的,映射的是一个Map,这个Map以这个对象的所有元素属性名作为键,其值为一个Set,这个Set内存储当前键的所有副作用函数。

这样每个副作用函数就精细地绑定到一个对象的一个元素上。

3、关于WeakMap

这里还需要说下WeakMap和Map的区别,其实这两者的映射关系是相同的,区分是WeakMap为弱引用。什么是弱引用呢,假设我们有个对象仅仅被Map引用,此时对象没有其他引用,这时对象是不会被垃圾回收器回收的,因为这个Map存在则对象的引用一直存在,但是如果这个对象仅仅被一个WeakMap引用,在无其他引用时会被垃圾回收器回收。即WeakMap不会影响垃圾回收器,如下代码可以很好解释:

const weakMap = new WeakMap()
const map = new Map()
(function () {
  const foo = {foo: 1}
  const bar = {bar: 1}

  map.set(foo, 1)
  weakMap.set(bar, 1)
})()

在立即执行函数内,两个对象foo被map引用,bar被weakMap引用,当执行完之后,bar会被回收掉,而foo依旧存在。

WeakMap还有一点即没有迭代器,无法像Map一样直接迭代遍历值,所以一般WeakMap常用于不会影响对象本身的映射,或者用于标记。

在Vue中使用WeakMap作为最外面的桶也很好理解,这不会导致对象被桶长期引用而无法被回收,桶并不会影响程序本身的执行。

4、分支切换与cleanup

在介绍分支切换之前可以先将上述代码做一个封装,将get拦截函数理中关于副作用的收集封装成track函数,将set函数中副作用函数的触发分装到trigger函数中,如下代码:

// 使用桶将所有的包含副作用函数的对象放入
  const bucket = new WeakMap()
  const obj = new Proxy(data, {
    get(target, key) {
      // 追踪函数
      track(target, key)
      return target[key]
    },
    set(target, key, newVal) {
      target[key] = newVal
      // 触发函数
      trigger(target, key)
      return true
    }
  })

  function track(target, key){
    // 没有副作用函数直接返回
    if(!activeEffect) return target[key]
    // 获取当前对象内的所有元素的Map
    let depsMap = bucket.get(target)
    // 当前对象还没有Map时新建一个
    if(!depsMap) {
      bucket.set(target, (depsMap = new Map()))
    }
    // 获取当前元素的所有副作用函数的Set
    let deps = depsMap.get(key)
    // 没有时新建并添加副作用函数
    if(!deps) depsMap.set(key, (deps = new Set()))
    deps.add(activeEffect)
  }

  function trigger(target, key) {
    const depsMap = bucket.get(target)
    if(!depsMap) return
    const deps = depsMap.get(key)
    deps && deps.forEach(fn => fn())
  } 
}

这样在关于Proxy的代理中就简单明了很多。

下面我们看一段代码:

const data = {ok: true, text: 'hello world'}
const obj = new Proxy(data, {/* 省略 */})

effect(() => {
  // 如果obj.ok为true则读取obj.text的值
  document.body.innerText = obj.ok? obj.text: 'not'
})

在我们写的响应式系统中,这个副作用函数会同时和obj.ok和obj.text关联,无论修改obj.ok还是修改obj.text都会触发这个副作用函数。但是仔细想想这真的有必要吗,其实是没有必要的。

在这段代码里如果obj.ok为true,这时这个副作用函数和obj.text是关联的,否则不关联,因为此时body的文本内容永远为not。

那怎么优化这一点呢,其实我们在上一篇文章中说过,一个响应式数据执行流程为:

  1. 修改obj.content的值,这会触发effect函数的执行 。
  2. 触发effect函数,这会获取obj.content的值。

之前在这里讲述过第二点,是否触发effect函数一定会获取obj.content的值,回答是肯定的,如果这里不获取obj.content的值则不需要建立响应式数据了。

我们根据这一点,在副作用函数执行之前将这个副作用函数从所有元素中删去,然后再执行副作用函数,如果副作用函数执行过程中需要读取当前对象的元素值,这会重新建立副作用函数。

这样我们代码需要增加三个部分,第一部分是cleanup用于从所有元素中删除副作用函数,第二个部分是在副作用函数中增加一个数组,用于记录和这个副作用函数相关的所有元素的Map,第三个部分是在track的时候将与该副作用关联的元素记录下来。

function cleanup(effectFn) {
    // 遍历包含副作用函数effctFn的集合
    for(let i=0;i<effectFn.deps.length;i++) {
      const deps = effectFn.deps[i]
      // 在集合中将effctFn副作用函数删去,在执行的时候会重新建立
      deps.delete(effectFn)
    }
    // 包含该副作用函数的集合目前为0
    effectFn.deps.length = 0
}

function effect(fn) {
    const effectFn = () => {
      activeEffect = effectFn
      //先清除再执行,自然就形成了分支切换
      cleanup(effectFn)
      fn()
    }
    effectFn.deps = []
    effectFn()
  }

function track(target, key){
    // 没有副作用函数直接返回
    if(!activeEffect) return target[key]
    // 获取当前对象内的所有元素的Map
    let depsMap = bucket.get(target)
    // 当前对象还没有Map时新建一个
    if(!depsMap) {
      bucket.set(target, (depsMap = new Map()))
    }
    // 获取当前元素的所有副作用函数的Set
    let deps = depsMap.get(key)
    // 没有时新建并添加副作用函数
    if(!deps) depsMap.set(key, (deps = new Set()))
    deps.add(key, activeEffect)

	// 这里将副作用函数相关的Map记录下来
    activeEffect.deps.push(deps)
  }

其实分支切换的思想很简单,就是在副作用函数执行前将当前副作用函数从关联的元素中全部删除,然后再执行副作用函数,在执行的时候如果读取了该元素,副作用函数又会重新关联,自然就形成了分支切换。

这样下来我们的响应式系统又完善了不少,其实还有不少的问题还没结局,比如嵌套的effect如何执行,如何给副作用函数做调度,是否会存在无限递归等问题,这会在后面讲解

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/502581.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

在CRA中配置别名路径并添加别名路径提示

写在前面&#xff1a; 使用React官方脚手架create-react-app[简称CRA]创建react项目&#xff1a;npx create-react-app 项目名称 一、配置别名路径 1.1 写在前面 目的&#xff1a;简化项目中的路径处理&#xff0c;和Vue项目中的类似。 参考文档&#xff1a;自定义CRA的默认…

MySQL基础(十二)数据类型精讲

1. MySQL中的数据类型 类型类型举例整数类型TINYINT、SMALLINT、MEDIUMINT、INT(或INTEGER)、BIGINT浮点类型FLOAT、DOUBLE定点数类型DECIMAL位类型BIT日期时间类型YEAR、TIME、DATE、DATETIME、TIMESTAMP文本字符串类型CHAR、VARCHAR、TINYTEXT、TEXT、MEDIUMTEXT、LONGTEXT枚…

一个日期类深度认识operator符号重载

一&#xff1a;概念 在以前的C语言的学习中,如果我们需要比较两个整数并返回它的结果可以直接用与之相关的符号。例如我们可以直接写成A>B或者A<B一类的&#xff0c;但是它的局限性很大&#xff0c;只能比较内置类型&#xff0c;因为计算可以直接转换成对应的汇编代码进…

如何通过国外主机租用服务提高网站SEO排名?

当今的互联网已经成为了商业和社交活动的主要场所之一。在这个快速变化的数字时代&#xff0c;网站的搜索引擎优化(SEO)排名对于任何企业的成功都至关重要。一个好的SEO排名能够帮助企业吸引更多的访客和潜在客户&#xff0c;增加业务的转化率。而国外主机租用服务可以帮助您优…

【C++学习】函数模板

模板的概念 模板就是建立通用的模具&#xff0c;大大提高复用性。 模板的特点&#xff1a; 模板不可以直接使用&#xff0c;它只是一个模型 模板的通用不是万能的 基本语法 C中提供两种模板机制&#xff1a;函数模板和类模板 函数模板作用&#xff1a; 建立一个通用函数&…

C++学习day--05 C++数据类型

1、项目需求&#xff1a;实现黑客攻击系统菜单打印 实现&#xff1a; #include <iostream> #include <Windows.h> int main( void ) { std::cout << "1. 网站 404 攻击 " << std::endl; std::cout << "2. 网站篡改攻击 …

实验四 基于PPTP的远程VPN实现【网络安全】

实验四 基于PPTP的远程VPN实现【网络安全】 前言推荐实验四 基于PPTP的远程VPN实现使用&#xff1a;配置CentOS PPTP服务端配置CentOS PPTP客户端常见问题浏览器无法打开网页 最后 前言 2023-5-7 23:10:12 以下内容源自《【网络安全】》 仅供学习交流使用 推荐 第27节 远程…

TCP三次握手/四次挥手

TCP三次握手/四次挥手 TCP的三次握手和四次挥手实质就是TCP通信的连接和断开。 三次握手 任何基于TCP的应用&#xff0c;在发送数据之前&#xff0c;都需要由TCP进行三次握手进行连接 握手流程&#xff1a; 三次握手原理 第1次握手&#xff1a;客户端发送一个带有SYN&#…

【蓝桥杯国赛真题26】Scratch队列练习 少儿编程scratch图形化编程 蓝桥杯省赛真题讲解

目录 scratch队列练习 一、题目要求 编程实现 二、案例分析 1、角色分析

2018年下半年软件设计师下午试题

试题四&#xff08;15分&#xff09; 给定一个字符序列Bb1b2….bn&#xff0c;其中bi∈{A,C,G,U}。B上的二级结构是一组字符对集合S{(bi,bj)},其中i,j∈{1,2,….,n}&#xff0c;并满足以下四个条件&#xff1a; &#xff08;1&#xff09;S中的每对字符是(A,U),(U,A),(C,G)和…

【网络】-- IP协议

应用层&#xff08;http、https&#xff09;&#xff1a; 数据的使用。传输层&#xff08;UDP、TCP&#xff09;&#xff1a;网络通讯的细节&#xff0c;将数据可靠的从A主机跨网络送到B主机。网络层&#xff08;IP&#xff09;&#xff1a;提供一种能力&#xff0c;将数据从A主…

如何让任何小程序都支持在windows系统中打开?

随着小程序的发展&#xff0c;出现了越来越多小程序在PC端打开的需求。很多程序员同行都想了解&#xff1a;小程序支持在windows系统、mac、统信UOS等桌面操作系统中打开吗&#xff1f; 答案当然是&#xff1a;可以&#xff01; 本文就基于作者自身的经验&#xff0c;给大家介…

IPWorks SSH 2022.0.8505 C++ Edition Crack

IPWorks SSH 2022.0.8505 C Edition 轻松将安全外壳 &#xff08;SSH&#xff09; 安全性集成到您的互联网应用程序中。IPWorks SSH 库包括支持 SSH 的客户端、服务器和代理组件&#xff0c;支持强 SSH 2.0 加密和高级加密。 SSH库 SSH 文件传输和通信 借助 IPWorks SSH&#x…

【嵌入式环境下linux内核及驱动学习笔记-(10-内核内存管理)】

目录 1、linux内核管理内存1.1 页1.2 区1.2.1 了解x86系统的内核地址映射区&#xff1a;1.2.2 了解32位ARM系统的内核地址映射区&#xff1a; 2、内存存取2.1 kmalloc2.1.1 kfree2.1.2 kzalloc 2.2 __get_free_page函数族2.2.1 free_page2.2.2 __get_free_pages()2.2.3 free_pa…

20230502 - 二叉树1 | 二叉树理论基础、二叉树的递归遍历

1、二叉树理论基础篇 二叉树可以链式存储&#xff0c;也可以顺序存储。 用数组来存储二叉树如何遍历的呢&#xff1f; 如果父节点的数组下标是 i&#xff0c;那么它的左孩子就是 i * 2 1&#xff0c;右孩子就是 i * 2 2。 深度优先遍历 前序遍历&#xff08;递归法&…

Android 页面滑动悬浮资源位动画+滑动监听解决方案

一、介绍 在日常业务开发过程中&#xff0c;我们有好多资源位悬浮在页面上&#xff0c;特别是电商以及促销页面&#xff0c;有些公司恨不得把整个页面像叠汉堡一样&#xff0c;一层一层加内容&#xff0c;目的是想让更多的人通过他们的资源来完成更便捷的操作。 但是资源是会覆…

HarmonyOS版的“抖音”长啥样?有图有真相

“鸿蒙系统实战短视频App 从0到1掌握HarmonyOS”系列课程是面向HarmonyOS实战的视频教程&#xff0c;该课程会通过构建一个真实的短视频App来向读者展示HarmonyOS的全过程。 本节将演示基于HarmonyOS短视频App的核心功能。通过了解该App的功能&#xff0c;也能初步对本课程的内…

C++——类和对象(4)

作者&#xff1a;几冬雪来 时间&#xff1a;2023年5月8日 内容&#xff1a;C类和对象内容讲解 目录 前言&#xff1a; 1.操作符重载&#xff08;续&#xff09;&#xff1a; 前置和后置&#xff1a; 日期减日期&#xff1a; <<操作符&#xff1a; 结尾&#xff…

顶级白帽黑客必备的十大黑客技术

1.熟悉Linux系统和命令行操作&#xff1a; Linux是黑客的基石&#xff0c;几乎所有黑客工具和技术都是在Linux平台上运行的&#xff0c;熟悉Linux系统和命令行操作是必须的。 2.掌握网络协议和TCP/IP模型&#xff1a; 了解TCP/IP模型、网络协议和通信流程是黑客攻击的基础&a…

Python-exe调用-控制台命令行执行-PyCharm刷新文件夹

文章目录 1.控制台命令行执行1.1.os.system()1.2.subprocess.getstatusoutput()1.3.os.popen() 2.PyCharm刷新文件夹3.作者答疑 1.控制台命令行执行 主要三种方式实现。 1.1.os.system() 它会保存可执行程序中的打印值和主函数的返回值&#xff0c;且会将执行过程中要打印的…