Vue3中Watch的同步和异步

news2025/1/14 18:04:33

在 Vue 3 中,watch 是一个强大的工具,适合监视响应式数据的变化并处理副作用逻辑。最近在做CodeReview的时候,发现了一些对watch使用上不太合理的地方,整理了一个类似的例子。

案例分析

先来看看例子:

<template>
  {{ dataList }}
</template>

<script setup lang="ts">
import { ref, watch } from "vue";

const dataList = ref([]);
const props = defineProps(["disableList""type""id"]);
watch(
  () => props.disableList,
  () => {
    // 基于disableList的逻辑非常复杂,它同步计算一个新列表
    const newList = getListFromDisabledList(dataList.value);
    dataList.value = newList;
  },
  { deep: true }
);
watch(
  () => props.type,
  () => {
    // 基于类型的逻辑非常复杂,同步计算新列表
    const newList = getListFromType(dataList.value);
    dataList.value = newList;
  }
);
watch(
  () => props.id,
  () => {
    // 从数据库拉取数据
    fetchDataList();
  },
  { immediate: true }
);
</script>

在这个例子中,dataList 在模板中渲染。更新 props.id 和初始化时,会异步从服务器获取 dataList。

更新 props.disableList 和 props.type 时,会同步计算新的 dataList。

代码逻辑流程图如下:

alt

乍一看,上面的代码可能没什么问题,但当不熟悉这方面的新同事接手时,问题就来了。

通常,在接手一个我们并不熟悉的业务领域时,我们需要找到一个起点。对于前端来说,这个起点肯定是浏览器中的渲染页面。从模版中我们可以知道 dataList 变量是核心所在,它有多个来源。

首先,服务器通过对 props.id 的监视进行异步更新。然后,通过对 props.disableList 和 props.type 的监视同步更新。

此时,不熟悉业务的同事如果收到要更新检索 dataList 逻辑的产品需求,就必须首先熟悉其多个来源背后的逻辑。

那么就要去看getListFromDisabledList、getListFromType例复杂的代码,分析清楚应该修改哪块才能满足产品要求。

不过,在实际操作中,当维护别人的代码(尤其是复杂的代码)时,我们一般不喜欢修改现有的代码,而是在上面添加自己的代码。更改他人的复杂代码很可能会引入错误(特别是有一些摸不着头脑的逻辑时),而我们也可能会因此造成生产事故。因此,我们通常的做法是添加另一个监视器,并在那里实现 dataList 的最新业务逻辑:

watch(
  () => props.xxx,
  () => {
    // Add the latest business logic
    const newList = getListFromXxx(dataList.value);
    dataList.value = newList;
  }
);

经过多次迭代后,这个 vue 文件就会变得杂乱无章,其中包含大量的观察语句,从而导致 "意大利面条代码"(Spaghetti Code 是一个编程术语,用于描述结构混乱、难以理解和维护的代码)。又或许这种编码风格可能是为了假装提高自己在团队中的价值,确保自己在团队中的地位,没有其他人敢碰这个复杂的代码。哈哈。

破局

上面这个例子,实际上是由于dataList的变更源过多引起的,而且里面还包含同步和异步两种。我们可以增加computed,把同步的变更整合到computed里,只保留异步的变更:

<template>
  {{ renderDataList }}
</template>

<script setup lang="ts">
import { ref, computed, watch } from "vue";

const props = defineProps(["disableList""type""id"]);
const dataList = ref([]);

const renderDataList = computed(() => {
  const newDataList = getListFromDisabledList(dataList.value);
  return getListFromType(newDataList);
});

watch(
  () => props.id,
  () => {
    fetchDataList();
  },
  {
    immediate: true,
  }
);
</script>

我们不再渲染 dataList 变量,而是渲染 renderDataList。代码逻辑流程图如下:

alt

当新的团队成员收到迭代 dataList 相关业务的产品需求时,由于我们的整个业务逻辑现在已经变成了一个线性序列,因此新的团队成员可以快速理清业务逻辑。

然后,根据产品的要求,他们可以决定是修改同步逻辑还是修改异步逻辑。以下是修改同步逻辑的演示:

const renderDataList = computed(() => {
  // 添加最新的处理逻辑
  const xxxList = getListFromXxx(dataList.value);
  const newDataList = getListFromDisabledList(xxxList);
  return getListFromType(newDataList);
});

总结

我们应该更多地使用 computed 来处理同步逻辑,将异步逻辑保留在 watch 中的方法。 computed 本身具备缓存特性,通过使用computed,我们可以减少状态的数量,因为computed是计算属性,是一个中间结果,是因变量不是自变量。

  • 滥用 watch 会导致代码难以维护: 当 watch 被用于处理多个复杂的同步和异步更新时,会导致代码变得杂乱无章,给维护和理解业务逻辑带来困难。

  • computed 和 watch 的适当使用可以提高代码质量: 将同步逻辑放入 computed 中,这样可以将业务逻辑线性化,使得代码更加清晰。同时,将异步逻辑保留在 watch 中,可以更好地区分和管理不同类型的更新。

  • 优化后的代码结构有助于新成员快速上手: 通过将同步逻辑集中在 computed 中,新成员可以快速定位需要修改的业务逻辑,提高开发效率和代码的可维护性。

  • 代码逻辑应该是线性和可追溯的: 代码逻辑应该像一个线程一样,从而使得业务逻辑的流向清晰明了,避免复杂的逻辑分叉和嵌套,这有助于代码的长期维护。

本文由 mdnice 多平台发布

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

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

相关文章

2024年10月13日(星期天)骑行知青堂

2024年10月13日 (星期天&#xff09;骑行知青堂&#xff0c;早8:30到9:00&#xff0c; 金殿水库大坝集合&#xff0c;9:00准时出发【因迟到者&#xff0c;骑行速度快者&#xff0c;可自行追赶偶遇。】 偶遇地点:金殿水库大坝集合 &#xff0c;家住东&#xff0c;南&#xff0…

华为Datacom考什么?要考几门?

华为认证细致地分为 HCIA、HCIP 和 HCIE 这三个具有显著差异的级别。 就在今天&#xff0c;让咱们一同分别针对改版后三个级别的 Datacom 考试展开全面且深入的介绍。 细致探讨看看它们各自具体考查哪些内容&#xff0c;究竟要通过考几门课程才能成功获取证书。 01、HCIA-Dat…

自动化测试中如何高效进行元素定位!

前言 在自动化测试中&#xff0c;元素定位是一项非常重要的工作。良好的元素定位可以帮助测试人员处理大量的测试用例&#xff0c;加快测试进度&#xff0c;降低工作负担。但是在实际的测试工作中&#xff0c;我们常常遇到各种各样的定位问题&#xff0c;比如元素定位失败、元…

Java的锁机制详解

在并发编程中&#xff0c;锁 是用于控制多个线程对共享资源进行访问的工具。Java提供了多种锁机制&#xff0c;从最基础的 synchronized 到高级的 ReentrantLock&#xff0c;这些锁帮助我们确保线程安全&#xff0c;并能有效避免数据竞争和死锁问题。 1. synchronized 关键字…

使用 Yarn 下载前端依赖报错

背景&#xff1a;尝试使用yarn下载前端项目依赖 运行&#xff1a; yarn install报错&#xff1a; 错误: 找不到或无法加载主类 install运行&#xff1a; yarn提示&#xff1a; 怀疑是否是 yarn 安装的问题&#xff1a; 执行&#xff1a;yarn -v 这个错误提示通常说明系…

Kubernetes(K8s)的简介

一、Kubernetes的简介 1 应用部署方式演变 在部署应用程序的方式上&#xff0c;主要经历了三个阶段&#xff1a; 传统部署&#xff1a;互联网早期&#xff0c;会直接将应用程序部署在物理机上 优点&#xff1a;简单&#xff0c;不需要其它技术的参与 缺点&#xff1a;不能为应…

《黑神话悟空:点燃朔州文旅之火》

《黑神话悟空&#xff1a;点燃朔州文旅之火》 一、黑神话悟空的魅力风暴 《黑神话&#xff1a;悟空》自上线以来&#xff0c;犹如一场魅力风暴席卷全球。这款以中国神话为背景的游戏&#xff0c;在美术设计和场景构建上大量取材中式古建&#xff0c;为玩家带来了一场视觉盛宴。…

求BMI python代码

问题&#xff1a; 肥胖的具体阶段采用更标准的方法来衡量&#xff0c;我们可以用BMI指数来判断&#xff0c;BMI也可称是体质指数&#xff0c;是国际上常用的衡量人体胖瘦程度以及是否健康的一个标准。 计算公式为&#xff1a;BMI体重&#xff08;kg&#xff09;身高&#xff…

记账本妙用:精准记录借还款,掌握财务脉搏

在日常生活中&#xff0c;无论是个人还是家庭&#xff0c;都难免会遇到借还款的情况。这些财务往来如同生活的细流&#xff0c;虽然每一笔可能不大&#xff0c;但汇聚起来却对财务状况有着不可忽视的影响。为了更好地掌握自己的财务脉搏&#xff0c;避免混乱和误解&#xff0c;…

C语言中的文件操作(二)

C语言中的文件操作&#xff08;一&#xff09;-CSDN博客https://blog.csdn.net/Xiaodao12345djs/article/details/142746010?spm1001.2014.3001.5501 四、文件的顺序读写 1、fputc (字符输出函数/写) 将一个字符写入文件中 #include <stdio.h>int main() {FILE* pf fo…

学习threejs,光晕效果

&#x1f468;‍⚕️ 主页&#xff1a; gis分享者 &#x1f468;‍⚕️ 感谢各位大佬 点赞&#x1f44d; 收藏⭐ 留言&#x1f4dd; 加关注✅! &#x1f468;‍⚕️ 收录于专栏&#xff1a;threejs gis工程师 文章目录 一、&#x1f340;前言二、&#x1f340;光晕效果实现1. ☘…

Linux环境变量及命令行参数

目录 一、环境变量的概念和基本命令 二、环境变量的组织结构及获取环境变量的方式 &#xff08;1&#xff09;组织结构 &#xff08;2&#xff09;获取环境变量 命令行第三个参数 通过第三方变量environ获取 通过系统调用getenv获取 三、命令行参数 一、环境变量的概念和…

ComfyUI 快速入门:从安装到使用的全面指南!

前言 在现代 AI 应用中&#xff0c;图像生成已成为一个热门话题。ComfyUI 是一个功能强大的工具&#xff0c;可以帮助用户轻松生成高质量的图像。本文将详细介绍如何在 Mac M系列&#xff08;我的电脑是M2的&#xff09; 和 Windows 上安装和使用 ComfyUI&#xff0c;从准备环…

Lumerical——Eigensolver Analysis

一、整体布局 上图显示的是“模式分析(MODE ANALYSIS )”窗口。①部分是模式列表 (MODE LIST)”,其中显示模式个数、反射率、传播损失、以及偏振方向。②部分显示的是计算参数;按照启动时设置,窗口左下角显示进行模拟计算用到的缺省参数。③部分包含模拟数据的“…

rpm软件包的制作方法

源码包制作成rpm包&#xff0c;可以在centos系列系统上平移 其他机器需要这个环境&#xff0c;把包发过去&#xff0c;就可以yum -y install new_pkg_name.rpm安装 比较方便 rpm包构建过程&#xff1a; 第一步&#xff1a;下载tar.gz源码包 在nginx官网下载自己准备用的版本&am…

查询效率提升近200倍!AntDB数据库助力医疗行业省级医保信息平台建设

导读&#xff1a; 本文介绍了在西北某省医保信息平台架构演进升级过程中&#xff0c;AntDB对于新建平台数据分析能力、数据同步能力、数据运维管理能力的提升。本次AntDB数据库多引擎能力的上线&#xff0c;解决了该医保系统数据同步不准确、数据查询结果不准确、复杂数据查询…

U盘数据丢失?这4款神器助你找回!

朋友们&#xff0c;今天咱们来聊聊U盘数据恢复工具&#xff1b;你是不是也遇到过U盘数据丢失的尴尬情况&#xff1f;别急&#xff0c;今天就让我来给你推荐几款好用的U盘数据恢复工具&#xff0c;并分享一下我的使用感受。 第一款&#xff1a;福昕数据恢复 直通车&#xff08;…

OBOO鸥柏:布局于为无人机展厅行产业提供LCD液晶显示终端

新华网快讯&#xff0c;于10月9日消息&#xff0c;有投资者在互动平台向OBOO鸥柏公司提问&#xff1a;您好&#xff01;目前有哪些工业/商用显示产品应用于无人机展厅展馆场景&#xff1f;能否着重介绍下贵司屏幕主要应用哪些品牌无人机数字化展厅展馆做保障&#xff1f; 鸥柏…

【ELKB】Kibana使用

搭建好ELKB后访问地址&#xff1a;http://localhost:5601 输入账号密码登录以后 左侧导航有home、Analysis、Enterprise search 、Observability、Security、Management home&#xff1a;首页Analysis&#xff1a;工具来分析及可视化数据Enterprise search&#xff1a;企业级搜…

原生USDC正式上线Sui

今天&#xff0c;标志着Sui生态的一个重要里程碑 — — 原生USDC现已正式在Sui主网上线。作为最广泛使用的稳定币之一&#xff0c;USDC为日益增长的Sui生态带来了稳定的价值传输和流动性。 随着Sui DeFi锁仓量&#xff08;TVL&#xff09;突破10亿美元&#xff0c;网络上需要更…