Vue.js---计算属性computed和lazy
4.6 计算属性computed和lazy
懒执行的effect:一般的effect一下子就执行了,但是懒加载effect是等需要的时候才会执行
这时我们通过在options中添加lazy属性来达到目的
function effect (fn , options = {}) {const effectFn = () => {// 调用clearup函数完成清除工作clearUp(effectFn);activeEffect = effectFn; // 在调用副作用函数之前将副作用函数压入栈中effectStack.push(effectFn);fn();// 执行完之后抛出,把activeEffect还原成原来的值effectStack.pop();activeEffect = effectStack[effectStack.length - 1];}// 将options挂在到fn上effectFn.options = options;// deps用来存储所有与该副作用函数相关联的依赖集合effectFn.deps = [];// 只有非lazy的时候才会执行副作用函数if(!options.lazy){effectFn(); }return effectFn}
那什么时候才执行这个函数呢?
因为我们前面返回了effectFn作为effect函数的返回值,那我们就可以手动调用该副作用函数了
const effectFn = effect(() => {console.log(obj.foo);} ,// 添加lazy{lazy : true})// 手动执行副作用函数effectFn();obj.foo++;obj.foo++;
现在我们要实现假设传递给effect函数的是getter函数,可以返回getter函数的值
const effectFn = effect(() => {obj.foo + obj.bar} ,// 添加lazy{lazy : true})// 手动执行副作用函数const value = effectFn();console.log(value);
我们希望输出的是 obj.foo + obj.bar的值
实现computed函数
// computed函数// 接收getter作为参数function computed (getter) {// 将gettwe作为副作用函数,创建一个lazy effectconst effectFn = effect( getter, {lazy : true})const obj = {// 对象的value是一个访问器属性// 只有当读取value的值时,才会执行effctFn并将其作为结果返回get value(){return effectFn()}}// 返回一个对象return obj;}// 开始使用计算属性const sumRes = computed(() => obj.foo + obj.bar)console.log(sumRes);//3
但是有一个bug。多次访问会导致effectFn多次计算
解决—对值进行缓存
function computed (getter) {// value用来缓存let value// dirty标志,标识是否需要重新计算let dirty = true// 将gettwe作为副作用函数,创建一个lazy effectconst effectFn = effect( getter, {lazy : true})const obj = {// 对象的value是一个访问器属性// 只有当读取value的值时,才会执行effctFn并将其作为结果返回get value(){// 只有true时才可以进行计算if(dirty){value = effectFn()// 将dirty设置为false,下一次直接使用缓存到value当中的值dirty = false;}return value}}// 返回一个对象return obj;}
无论我们执行了多少次sumRes都不会重新执行啦,直接取value里面的值
但是我们如果修改obj.foo的值,我们发现并没有响应的修改最后的sumRes
get value(){// 只有true时才可以进行计算if(dirty){value = effectFn()// 将dirty设置为false,下一次直接使用缓存到value当中的值dirty = false;}return value}
dirty = false;这便是原因
解决:当值发生变化的时候,改变dirty的值就可以啦,这时我们就要使用scheduler选项
// computed函数// 接收getter作为参数function computed (getter) {// value用来缓存let value// dirty标志,标识是否需要重新计算let dirty = true// 将gettwe作为副作用函数,创建一个lazy effectconst effectFn = effect( getter, {lazy : true,scheduler(){dirty = true}})const obj = {// 对象的value是一个访问器属性// 只有当读取value的值时,才会执行effctFn并将其作为结果返回get value(){// 只有true时才可以进行计算if(dirty){value = effectFn()// 将dirty设置为false,下一次直接使用缓存到value当中的值dirty = false;}return value}}// 返回一个对象return obj;}// 开始使用计算属性const sumRes = computed(() => obj.foo + obj.bar)console.log(sumRes.value);//4obj.foo++console.log(sumRes.value);//5
我们的计算属性已经趋于完美了,但是还有一个问题
effect(() => {console.log(sumRes.value)})obj.foo++
我们希望的是在完成 obj.foo++之后是可以重新渲染的,但是我们发现并没有
其实这是一个典型的effect嵌套,一个计算属性的内部有effect,并且它是懒执行的,只有当真正读取计算属性的值才会执行。
- 例如,假设我们有一个计算属性
fullName
,它依赖于响应式数据firstName
和lastName
。只有当我们读取fullName
的值时,才会执行计算fullName
的effect
。
但是getter里面访问的响应式数据只会把computed内部的effect收集为依赖。
而当把计算属性用于另一个effect时,就会发生effect嵌套,外层的effect不会被内层的effect所收集
- 例如,假设我们有一个
effect
函数updateUI
,它读取fullName
。这时,updateUI
的effect
和fullName
的effect
就会嵌套在一起。
解决—当读取计算属性的值时,我们可以手动调用track函数进行追踪,当计算属性发生变化的时候,手动调用trigger触发响应
- 当读取计算属性的值时,手动调用
track
函数,将当前effect
(如updateUI
)添加到计算属性的依赖列表中。 - 这样,当计算属性的依赖发生变化时,可以触发当前
effect
。 - 当计算属性的值发生变化时,手动调用
trigger
函数,通知所有依赖该计算属性的effect
进行更新。 - 这样,外层的
effect
(如updateUI
)会在计算属性(如fullName
)的依赖发生变化时被触发。
完整代码:
<script setup>let activeEffect;// effect栈const effectStack = [];function effect (fn , options = {}) {const effectFn = () => {// 调用clearup函数完成清除工作clearUp(effectFn);activeEffect = effectFn; // 在调用副作用函数之前将副作用函数压入栈中effectStack.push(effectFn);// 将effect的值存储起来const res = fn();// 执行完之后抛出,把activeEffect还原成原来的值effectStack.pop();activeEffect = effectStack[effectStack.length - 1];// 返回resreturn res;}// 将options挂在到fn上effectFn.options = options;// deps用来存储所有与该副作用函数相关联的依赖集合effectFn.deps = [];// 只有非lazy的时候才会执行副作用函数if(!options.lazy){effectFn(); }return effectFn}const bucket = new WeakMap();const data = { foo : 2 , bar : 2 }; // 确保所有属性都已定义const obj = new Proxy(data, {get(target, key){track(target , key);return target[key];},set(target, key, newVal){target[key] = newVal;trigger(target , key , newVal);}});// 追踪变化function track(target , key){if(!activeEffect){return;}// 根据tartget取来的depsMap,它是一个map类型let depsMap = bucket.get(target);// 如果不存在if(!depsMap){// 创建一个bucket.set(target, (depsMap = new Map()));}// 根据key取来的deps,它是一个set类型let deps = depsMap.get(key);// 如果不存在if(!deps){// 创建一个depsMap.set(key, (deps = new Set()));}deps.add(activeEffect); // 添加当前活跃的副作用函数activeEffect.deps.push(deps);}
// 触发变化:处理调度逻辑function trigger(target, key, newVal) {const depsMap = bucket.get(target);if (!depsMap) {return;}const effects = depsMap.get(key);// 开始执行const effectsToRun = new Set(effects);effects && effects.forEach(effectFn => {if(activeEffect !== effectFn){effectsToRun.add(effectFn);}});effectsToRun.forEach(effectFn =>{if(effectFn.options.scheduler){effectFn.options.scheduler(effectFn);}else {effectFn();}});// effects && effects.forEach(fn => fn()); // 只触发与键相关的副作用函数}// 清除函数function clearUp (effectFn){// 遍历然后进行删除for(let i = 0 ; i < effectFn.deps.length ; i++){const deps = effectFn.deps[i];// 移除deps.delete(effectFn);}// 最后重置effectFn.deps数组effectFn.deps.length = 0;}// effect(() => {// console.log(obj.foo);// })// obj.foo ++;// obj.foo++;// 连续执行同一代码,只饭后最后一次计算的结果// 定义一个任务集合set:可以进行去重操作const jobQueue = new Set();// 使用promise.resolve()创建一个promise实例:将一个任务添加到微任务队列当中const p = Promise.resolve();// 开始刷新队列let isFlushing = false;function flushJob(){// 如果队列正在被刷新->returnif(isFlushing){return;}// 没有刷新,进行刷新操作isFlushing = true;p.then(() => {jobQueue.forEach(job => job());}).finally(() => {// 结束后重置isFlushing = false;})}// computed函数// 接收getter作为参数function computed (getter) {// value用来缓存let value// dirty标志,标识是否需要重新计算let dirty = true// 将gettwe作为副作用函数,创建一个lazy effectconst effectFn = effect( getter, {lazy : true,scheduler(){// 设置value进行triggertrigger(obj , 'value');dirty = true}})const obj = {// 对象的value是一个访问器属性// 只有当读取value的值时,才会执行effctFn并将其作为结果返回get value(){// 只有true时才可以进行计算if(dirty){value = effectFn()// 将dirty设置为false,下一次直接使用缓存到value当中的值dirty = false;}// 读取value,进行追踪track(obj , 'value')return value}}// 返回一个对象return obj;}// 开始使用计算属性const sumRes = computed(() => obj.foo + obj.bar)console.log(sumRes.value);obj.foo++console.log(sumRes.value);effect(() => {console.log(sumRes.value)})obj.foo</script>
总结:
computed属性的实现我们首先用到了懒加载effect,需要使用的时候才使用。因为我们前面返回了effectFn作为effect函数的返回值,那我们就可以手动调用该副作用函数了。接着我们实现了computed属性,我们是传入一个getter函数和懒加载属性,为了解决多次访问会导致effectFn多次计算,我们需要缓存value。但是我们如果修改obj.foo的值,我们发现并没有响应的修改最后的sumRes,那是因为dirty并没有在修改值之后被修改为true,所以我们就要使用scheduler选项,在 effect
的 options
中添加 scheduler
,当计算属性的依赖发生变化时,将 dirty
设为 true
,以便下次读取 value
时重新计算。为了解决effect嵌套的问题,我们进行了手动追踪和触发
相关文章:
Vue.js---计算属性computed和lazy
4.6 计算属性computed和lazy 懒执行的effect:一般的effect一下子就执行了,但是懒加载effect是等需要的时候才会执行 这时我们通过在options中添加lazy属性来达到目的 function effect (fn , options {}) {const effectFn () > {// 调用clearup函数…...

找客户的app
找客户的 app 在竞争激烈的商业环境中,找客户的 APP 成为企业拓展业务的利器。 微拓客 APP,集智能获客、营销素材、客户管理于一体。支持关键词、附近客源等多方式采集,覆盖 300 行业;一键采集客源,一键导出到通讯录…...
HarmonyOS学习
个人简介 👨💻个人主页: 魔术师 📖学习方向: 主攻前端方向,正逐渐往全栈发展 🚴个人状态: 研发工程师,现效力于政务服务网事业 🇨🇳人生格言&…...

金融合规革命:R²AIN SUITE 如何重塑银行业务智能
一、市场发展背景与核心驱动因素 信息过载:单家银行年均新增监管文件大量增加,人工解读效率极低。 客户体验升级:高净值客户期待“724小时专业级响应”,但客户经理难以实时掌握数百款产品动态。 风险防控:传…...
Spring 框架核心机制深度解析【AI模型互搏生成】
Spring 框架核心机制深度解析(玩转开源代码) 一、Bean 生命周期全流程剖析 1.1 核心生命周期阶段 以下是 Spring Bean 生命周期核心阶段的配图,结合实际流程图示清晰展现每一步执行顺序及扩展点挂载位置。 🧬Spring Bean 生命周…...
Linux信号的保存
Linux系统中信号的保存涉及内核为每个进程维护的数据结构,确保信号在产生后、处理前被正确记录和管理。以下是详细的解释: 1. 信号的基本概念 信号(Signal):用于通知进程发生了特定事件的异步通知机制,如…...

论文阅读:Self-Collaboration Code Generation via ChatGPT
地址:Self-Collaboration Code Generation via ChatGPT 摘要 尽管大型语言模型(LLMs)在代码生成能力方面表现出色,但在处理复杂任务时仍存在挑战。在现实软件开发中,人类通常通过团队协作来应对复杂任务,…...

2025年PMP 学习十五 第10章 项目资源管理
2025年PMP 学习十五 第10章 项目资源管理 序号过程过程组1规划沟通管理规划2管理沟通执行3监控沟通监控 项目沟通管理包括为确保项目的信 息及时且恰当地规划、收集、生成、发布、存储、检索、管理、控制、监 警和最终处理所需的过程; 项目经理绝大多数时间都用于与…...

如何使用易路iBuilder智能体平台快速安全深入实现AI HR【实用帖】
随着企业组织经营对降本、增效、提质的需求日益迫切,越来越多企业启动人力资源数智化转型战略。而在AI战略实际推进过程中,企业组织往往在选型、搭建、使用、管控等问题上面临困惑: 如何快速、低成本接入AI能力,实现人力资源管理…...
高效管理嵌套Git仓库:三合一脚本解决方案
背景介绍 在大型软件开发项目中,我们经常会遇到Git仓库嵌套的情况(即一个Git仓库中包含其他Git子仓库)。传统的手动管理方式效率低下,容易出错。本文将介绍三个精心设计的Shell脚本,帮助开发者高效扫描、克隆和更新嵌套Git仓库。 脚本功能解析 1. clone_dep_repo.sh - …...

免费实用的远程办公方案
假如你需要快速检索出远程电脑文件并下载? 假如你需要访问远程电脑的共享文件夹? 假如你需要访问远程电脑的USB设备,例如软件加密狗、调试器、固件烧录器、U盘等? 本篇文章能够解决以上痛点。 这个方案非常实用,也很…...

【springboot项目服务假死、内存溢出问题排查】
问题现象:springboot服务A刚启动时正常,但运行几个小时后就会接口请求无响应,但服务器网络、磁盘I/O和CPU都没有出现爆满的情况,且A服务日志没有异常报错。 线上SpringBoot假死现象 SpringBoot应用会出现无法访问的情况。具体的表…...

Java 线程状态详解:从创建到销毁的完整旅途
前言 在 Java 多线程编程中,线程的状态管理是理解并发逻辑的核心。本文将用通俗的语言和代码示例,解析线程的6种状态及其转换条件,助你彻底掌握线程的生命周期。 一、线程的6种状态 状态含义NEW线程对象已创建,但未启动…...

操作系统|| 虚拟内存页置换算法
题目 写一个程序来实现 FIFO 和 LRU 页置换算法。首先,产生一个随机的页面引用序列,页面数从 0~9。将这个序列应用到每个算法并记录发生的页错误的次数。实现这个算法时要将页帧的数量设为可变。假设使用请求调页。可以参考所示的抽象类。 抽象类&…...

Maven 项目构建时编译错误问题排查与解决
1. 问题描述 Maven 项目执行命令 mvn clean package 时出现编译错误,如下图所示 2. 问题分析 由于是源码编译错误,于是通过查看项目 pom.xml 文件,得到项目源码使用的 Java 版本为 21 <project xmlns"http://maven.apache.org/P…...
5 Celery多节点部署
一、多节点部署架构设计 1.1 典型生产环境拓扑 #mermaid-svg-NjPQBLvUUsBc24uk {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-NjPQBLvUUsBc24uk .error-icon{fill:#552222;}#mermaid-svg-NjPQBLvUUsBc24uk .error…...
FPGA:Lattice的FPGA产品线以及器件选型建议
本文将详细介绍Lattice Semiconductor的FPGA产品线,帮助你了解各系列的特点和适用场景,以便更好地进行选型。Lattice以低功耗、小尺寸和高性能为核心,产品覆盖低中端市场,广泛应用于通信、计算、工业、汽车、消费电子、嵌入式视觉…...

安全生产调度管理系统的核心功能模块
安全生产调度管理系统是运用现代信息技术构建的智能化管理平台,旨在实现生产安全风险的全面管控和应急资源的优化调度。该系统通过整合物联网、大数据、人工智能等前沿技术,建立起覆盖风险监测、预警预测、指挥调度、决策支持的全链条安全管理体系。 一…...
R语言学习--Day03--数据清洗技巧
在一般情况下,我们都是在数据分析的需求前提下去选择使用R语言。而实际上,数据分析里,百分之八十的工作,都是在数据清洗。并不只是我们平时会提到的异常值处理或者是整合格式,更多会涉及到将各种各样的数据整合&#x…...

Linux进程信号(一)之信号的入门
文章目录 信号入门1. 生活角度的信号2. 技术应用角度的信号3. 注意4. 信号概念5.用kill -l命令可以察看系统定义的信号列表6. 信号处理常见方式 信号入门 1. 生活角度的信号 你在网上买了很多件商品,再等待不同商品快递的到来。但即便快递没有到来,你也…...

基于springboot+vue的机场乘客服务系统
开发语言:Java框架:springbootJDK版本:JDK1.8服务器:tomcat7数据库:mysql 5.7数据库工具:Navicat12开发软件:eclipse/myeclipse/ideaMaven包:Maven3.3.9 系统展示 用户管理 航班信…...

基于SpringBoot的房屋租赁管理系统
作者:计算机学姐 开发技术:SpringBoot、SSM、Vue、MySQL、JSP、ElementUI、Python、小程序等,“文末源码”。 专栏推荐:前后端分离项目源码、SpringBoot项目源码、Vue项目源码、SSM项目源码、微信小程序源码 精品专栏:…...
redis数据结构-11(了解 Redis 持久性选项:RDB 和 AOF)
了解 Redis 持久性选项:RDB 和 AOF Redis 提供了多个持久性选项,以确保数据持久性并防止在服务器发生故障或重启时丢失数据。了解这些选项对于为您的特定使用案例选择正确的策略、平衡性能和数据安全至关重要。本章节将深入探讨 Redis 中的两种主要持久…...

STM32外设AD/DA-基础及CubeMX配置
STM32外设AD/DA-基础及CubeMX配置 一,什么是AD/DA二,基础概念1,模拟 vs 数字2,AD转换1,分辨率 (Resolution)2,参考电压 (Reference Voltage, Vref)3,采样率 (Sampling Rate) 3,DA转换…...

React Native简介
React Native 是由 Meta(原 Facebook)开源的跨平台移动应用开发框架,基于 React 和 JavaScript,允许开发者使用同一套代码库构建 iOS 和 Android 原生应用。通过 JavaScript 调用原生组件实现高性能渲染。 跨平台开发 共享 80%-9…...
微服务如何实现服务的高并发
高并发的常见指标 响应时间吞吐量每秒查询率QPS并发用户数 高并发是分布式系统架构设计必须的考虑因素 具体实现方案粗略分两种: 垂直扩展 硬件升级方向 处理器:增加CPU核数(如升级至32核以上)或采用更高主频的CPU存储设备…...

GCC 使用说明
参数 -fPIC ppc_85xx-gcc -shared -fPIC liberr.c -o liberr.so -fPIC 作用于编译阶段,告诉编译器产生与位置无关代码(Position-Independent Code), 则产生的代码中,没有绝对地址,全部使用相对地址,故而代码可以被加…...
码蹄集——圆包含
MT1181 圆包含 输入2个圆的圆心的坐标值(x,y)和半径,判断断一个圆是否完全包含另一个圆,输出YES或者NO。另:内切不算做完全包含。 格式 输入格式:输入整型,空格分隔。 每行输入一组…...
CSR、SSR与ISR的奇妙之旅
网页渲染三剑客:CSR、SSR与ISR的奇妙之旅 三种渲染方式的核心本质 CSR(客户端渲染)让浏览器成为"厨师",SSR(服务器端渲染)让服务器担任"厨师",而ISR(增量静态再生)则是一位兼具"提前备餐"和"即时烹饪"能力的"超级厨师"…...

Verilog HDL 语言整理
Verilog HDL 语言 Verilog HDL 简介 硬件描述语言Hardware Description Language是一种用形式化方法即文本形式 来描述和设计数字电路和数字系统的高级模块化语言 Verilog HDL(Hardware Description Language)是一种硬件描述语言,用于建模…...