Vue3中computed和watch的区别

news2025/6/8 22:33:13

文章目录

  • 前言
    • 🔍 一、`computed` vs `watch`
      • ✅ 示例对比
        • 1. `computed` 示例(适合模板绑定、衍生数据)
        • 2. `watch` 示例(副作用,如调用接口)
    • 🧠 二、源码实现原理(简化理解)
      • 1. `computed` 原理
      • 2. `watch` 原理
    • 📌 三、使用建议
    • 扩展:
    • 🧠 四、Vue 3 响应式系统核心:`effect` / `track` / `trigger`
      • 1. `effect(fn)`:响应式副作用收集器
      • 2. `track(target, key)`:依赖追踪
      • 3. `trigger(target, key)`:依赖触发
      • 🔁 总结:响应式机制流程
    • 🔁 五、`watchEffect` 是什么?
      • 🌟 特点:
      • 📦 内部工作机制(简化版)
    • 🧪 应用场景对比
    • ✅ 实战案例:watch vs watchEffect
      • `watch` 示例(明确监听)
      • `watchEffect` 示例(更简洁)


前言

Vue 3 中 computedwatch区别源码实现逻辑(Composition API 版本)。


🔍 一、computed vs watch

项目computedwatch
类型派生状态(缓存)响应式副作用
用途根据已有响应式变量派生出新数据监听某个响应式数据的变化后执行副作用逻辑
是否缓存✅ 是❌ 否
返回值Ref(值类型)void(返回值无意义)
使用场景显示用、模板绑定API 调用、定时器、调试、数据同步等

✅ 示例对比

1. computed 示例(适合模板绑定、衍生数据)
const price = ref(100)
const tax = ref(0.1)

const total = computed(() => price.value * (1 + tax.value))
// total.value = 110
2. watch 示例(副作用,如调用接口)
watch(price, (newVal, oldVal) => {
  console.log(`价格变化:${oldVal} -> ${newVal}`)
  // 比如向服务器汇报价格变动
})

🧠 二、源码实现原理(简化理解)

1. computed 原理

  • Vue 内部使用了 effectRef 实现 computed
  • 默认是 lazy effect,只有在访问 .value 时才执行计算。
  • Vue 自动追踪依赖(依赖收集),只在依赖变化时重新执行。
function computed(getter) {
  const result = ref()
  const runner = effect(getter, {
    lazy: true,
    scheduler: () => {
      dirty = true
    }
  })

  return {
    get value() {
      if (dirty) {
        result.value = runner()
        dirty = false
      }
      return result.value
    }
  }
}

关键点:缓存 + 懒执行 + 自动依赖追踪


2. watch 原理

  • watch 是对 effect 的进一步封装。
  • 内部会递归地跟踪传入的响应式数据,建立响应关系。
  • 依赖变动时,会执行用户提供的回调函数。
function watch(source, cb) {
  let oldValue = source()

  effect(() => {
    const newValue = source()
    if (newValue !== oldValue) {
      cb(newValue, oldValue)
      oldValue = newValue
    }
  })
}

关键点:不缓存 + 只执行副作用 + 深层依赖可配置


📌 三、使用建议

需求推荐
只用来展示(派生数据)computed
需要执行副作用(发请求、console.log、同步)watch
复杂对象监听watch 配合 deep: true
多个数据合并监听watch([a, b])

扩展:

  • watchEffect 的解释
  • 或深入理解 Vue 的响应式核心 effecttracktrigger 机制

继续深入讲解 Vue 3 响应式系统的核心:effecttracktrigger,并说明 watchEffect 是如何工作的。


🧠 四、Vue 3 响应式系统核心:effect / track / trigger

1. effect(fn):响应式副作用收集器

  • 用于包裹副作用函数(如 computed 的 getter 或 watch 的监听函数)。
  • 在函数中读取响应式数据时,会被自动“追踪依赖”。
effect(() => {
  console.log(state.count)  // 依赖 `state.count`
})

2. track(target, key):依赖追踪

  • 当你读取响应式对象的属性时,会触发 track
  • track 会记录当前活跃的 effect,让这个 effect 依赖这个属性。
// 简化版
function track(target, key) {
  if (!activeEffect) return
  let depsMap = targetMap.get(target)
  if (!depsMap) {
    depsMap = new Map()
    targetMap.set(target, depsMap)
  }
  let deps = depsMap.get(key)
  if (!deps) {
    deps = new Set()
    depsMap.set(key, deps)
  }
  deps.add(activeEffect)
}

3. trigger(target, key):依赖触发

  • 当你修改响应式数据时,会触发 trigger
  • 它会找到依赖这个属性的所有 effect,然后重新执行它们。
function trigger(target, key) {
  const depsMap = targetMap.get(target)
  if (!depsMap) return
  const effects = depsMap.get(key)
  effects.forEach(effect => effect())
}

🔁 总结:响应式机制流程

  1. effect(fn) 执行时读取响应式数据 → track 收集依赖
  2. 数据变化时 → trigger 调用依赖的 effect → 重新执行

🔁 五、watchEffect 是什么?

🌟 特点:

  • 类似 computed,但用于副作用。
  • 自动收集依赖,无需显式声明。
  • 依赖变更时自动重新执行副作用函数。
watchEffect(() => {
  console.log(`count 是 ${count.value}`)
})

📦 内部工作机制(简化版)

function watchEffect(effectFn) {
  const runner = effect(effectFn)
  return () => stop(runner)
}
  • watchEffect 调用时立即执行 effectFn
  • 响应式依赖被收集。
  • 当依赖变更时,会重新调用 effectFn
  • 返回的函数可用于停止监听。

🧪 应用场景对比

需求场景推荐工具
计算派生值computed
副作用 + 明确监听字段watch
副作用 + 自动依赖收集watchEffect

✅ 实战案例:watch vs watchEffect

watch 示例(明确监听)

watch(() => user.id, (newId) => {
  fetchUserDetail(newId)
})

watchEffect 示例(更简洁)

watchEffect(() => {
  fetchUserDetail(user.id)
})

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

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

相关文章

学习STC51单片机30(芯片为STC89C52RCRC)

每日一言 当你感到疲惫时,正是成长的关键时刻,再坚持一下。 IIC协议 是的,IIC协议就是与我们之前的串口通信协议是同一个性质,就是为了满足模块的通信,其实之前的串口通信协议叫做UART协议,我们千万不要弄…

Python-进程

进程 简介 操作系统分配资源的基本单位 创建 依赖 依赖模块 multiprocessing 中的 Process 语法 Process(group[,target[,name[,args[,kwargs]]]]) target:如果传递了函数的引用,这个子进程就执行这里的代码args:元组的方式传递&#x…

Paraformer分角色语音识别-中文-通用 FunASR demo测试与训练

文章目录 0 资料1 Paraformer分角色语音识别-中文-通用1 模型下载2 音频识别测试3 FunASR安装 (训练用)4 训练 0 资料 https://github.com/modelscope/FunASR/blob/main/README_zh.md https://github.com/modelscope/FunASR/blob/main/model_zoo/readm…

对抗反爬机制的分布式爬虫自适应策略:基于强化学习的攻防博弈建模

在大数据时代,数据的价值不言而喻。网络爬虫作为获取数据的重要工具,被广泛应用于各个领域。然而,随着爬虫技术的普及,网站为了保护自身数据安全和服务器性能,纷纷采取了各种反爬机制。这就使得爬虫与反爬虫之间形成了…

手写muduo网络库(一):项目构建和时间戳、日志库

引言 本文作为手写 muduo 网络库系列开篇,聚焦项目基础框架搭建与核心基础工具模块设计。通过解析 CMake 工程结构设计、目录规划原则,结合时间戳与日志系统的架构,为后续网络库开发奠定工程化基础。文中附完整 CMake 配置示例及模块代码。 …

14-Oracle 23ai Vector Search 向量索引和混合索引-实操

一、Oracle 23ai支持的2种主要的向量索引类型: 1.1 内存中的邻居图向量索引 (In-Memory Neighbor Graph Vector Index) HNSW(Hierarchical Navigable Small World :分层可导航小世界)索引 是 Oracle AI Vector Search 中唯一支持的内存邻居图向量索引类…

Web前端基础:JavaScript

1.JS核心语法 1.1 JS引入方式 第一种方式&#xff1a;内部脚本&#xff0c;将JS代码定义在HTML页面中 JavaScript代码必须位于<script></script>标签之间在HTML文档中&#xff0c;可以在任意地方&#xff0c;放置任意数量的<script></script>一般会把…

基于AWS Serverless架构:零运维构建自动化SEO内容生成系统

作者&#xff1a;[Allen] 技术专栏 | 深度解析云原生SEO自动化 在流量为王的时代&#xff0c;持续产出高质量SEO内容成为技术运营的核心痛点。传统方案面临开发成本高、扩展性差、关键词响应滞后三大难题。本文将分享如何用AWS Serverless技术栈&#xff0c;构建一套零服务器运…

电镀机的阳极是什么材质?

知识星球&#xff08;星球名&#xff1a;芯片制造与封测技术社区&#xff0c;点击加入&#xff09;里的学员问&#xff1a;电镀的阳极有什么讲究&#xff1f;什么是可溶性阳极和非可溶性阳极&#xff1f; 什么是可溶性阳极与非可溶性阳极&#xff1f; 可溶性阳极 阳极本身就是…

vscode调试deepspeed的方法之一(无需调整脚本)

现在deepspeed的脚本文件是&#xff1a; # 因为使用 RTX 4000 系列显卡时&#xff0c;不支持通过 P2P 或 IB 实现更快的通信宽带&#xff0c;需要设置以下两个环境变量 # 禁用 NCCL 的 P2P 通信&#xff0c;以避免可能出现的兼容性问题 export NCCL_P2P_DISABLE"1" …

Codeforces Round 509 (Div. 2) C. Coffee Break

题目大意&#xff1a; 给你n、m、d n为元素个数,m为数列长度,d为每个元素之间的最短间隔 问最少需要多少个数列可以使得元素都能装进数列&#xff0c;并且满足每个元素之间的间隔大于等于d 核心思想 使用贪心的思想&#xff0c;将元素的大小进行排序&#xff0c;问题出在必…

榕壹云健身预约系统:多门店管理的数字化解决方案(ThinkPHP+MySQL+UniApp实现)

随着全民健身热潮的兴起&#xff0c;传统健身房在会员管理、课程预约、多门店运营等方面面临诸多挑战。针对这一需求&#xff0c;我们开发了一款基于ThinkPHPMySQLUniApp的榕壹云健身预约系统&#xff0c;为中小型健身机构及连锁品牌提供高效、灵活的数字化管理工具。本文将详细…

QUIC——UDP实现可靠性传输

首先我们要知道TCP存在什么样的痛点问题 TCP的升级很困难TCP建立连接的延迟网络迁移需要重新建立连接TCP存在队头阻塞问题 QUIC就是为了解决以上的问题而诞生了, 下面我会介绍QUIC的一些特性和原理 QUIC对比TCP优势: 握手建连更快 QUIC内部包含了TLS, 它在自己的帧会携带TL…

快速上手shell脚本运行流程控制

一、条件运行流程控制 1.if单分支结构 #!/bin/bash if [ 条件 ] then动作1动作2... fi 2.if双分支结构 ​ #!/bin/bash if [ 条件 ] then动作1动作2... else动作1动作2... fi​ 3.if多分支结构 二、循环运行流程控制 1.无判定for循环 给网卡一键添加5个IP 2.判断循环 while…

10.Linux进程信号

1. 理解信号 信号VS信号量 老婆&#xff1a;老婆饼-》没有任何关系&#xff01;信号&#xff1a;闹钟&#xff0c;上课铃声&#xff0c;脸色...人-》进程&#xff1b;信号中断人正在做的事&#xff0c;是一种事件的异步通知机制&#xff1b; 我们自习一会&#xff0c;等张三回…

机器学习基础(四) 决策树

决策树简介 决策树结构&#xff1a; 决策树是一种树形结构&#xff0c;树中每个内部节点表示一个特征上的判断&#xff0c;每个分支代表一个判断结果的输出&#xff0c;每个叶子节点代表一种分类结果 决策树构建过程&#xff08;三要素&#xff09;&#xff1a; 特征选择 选…

CentOS 7如何编译安装升级gcc至7.5版本?

CentOS 7如何编译安装升级gcc版本? 由于配置CentOS-SCLo-scl.repo与CentOS-SCLo-scl-rh.repo后执行yum install -y devtoolset-7安装总是异常&#xff0c;遂决定编译安装gcc7.5 # 备份之前的yum .repo文件至 /tmp/repo_bak 目录 mkdir -p /tmp/repo_bak && cd /etc…

为什么React列表项需要key?(React key)(稳定的唯一标识key有助于React虚拟DOM优化重绘大型列表)

文章目录 1. **帮助 React 识别列表项的变化**2. **性能优化**3. **避免组件状态混乱**4. **为什么使用 rpid 作为 key**5. **不好的做法示例**6. **✅ 正确的做法** 在 React 中添加 key{item.rpid} 是非常重要的&#xff0c;主要有以下几个原因&#xff1a; 1. 帮助 React 识…

飞牛云一键设置动态域名+ipv6内网直通访问内网的ssh服务-家庭云计算专家

IPv6访问SSH的难点与优势并存。难点主要体现在网络环境支持不足&#xff1a;部分ISP未完全适配IPv6协议&#xff0c;导致客户端无法直接连通&#xff1b;老旧设备或工具&#xff08;如Docker、GitHub&#xff09;需额外配置才能兼容IPv6&#xff0c;技术门槛较高&#xff1b;若…

Java高级 | 【实验七】Springboot 过滤器和拦截器

隶属文章&#xff1a;Java高级 | &#xff08;二十二&#xff09;Java常用类库-CSDN博客 系列文章&#xff1a;Java高级 | 【实验一】Springboot安装及测试 |最新-CSDN博客 Java高级 | 【实验二】Springboot 控制器类相关注解知识-CSDN博客 Java高级 | 【实验三】Springboot 静…