Vue 响应式的本质

news/2024/12/21 20:44:05/

在 Vue 开发中,最容易出问题的地方往往是对 Vue 响应式系统的误解。

响应式的核心不在于“数据一变,页面就刷新”,这是表象。真正的本质是数据和函数的关联

当数据与函数关联后,数据的变化会触发相应函数的重新运行。这里要明确两个关键点:哪些数据哪些函数关联了呢?

哪些数据会被关联?

1、函数中使用的数据:只有当函数用到了某个数据时,这个数据才与函数关联。例如,读取对象中的某个属性。

2、读取或使用的数据是响应式数据:只有 ref 或 reactive 包裹的数据才具备响应式特性,普通数据是不会触发关联函数运行的。

哪些函数会被监控?

被监控的函数,在 vue2 是放在 Wacther 内部的,在 vue3 是放在 effect 内部的。

1、render:渲染函数,也就解释了为什么数据一变,页面刷新,因为 render 是被监控的。

2、watchEffect:自动依赖追踪,数据变化时,函数会自动重新执行。

3、watch:显式地监听指定的数据变化,并执行相应回调。

4、computed:计算属性,它是基于响应式数据计算得来的,数据变化会重新计算值。

因此,只要满足这两点:响应式数据和被监控的函数,数据变化时,函数就会重新运行,页面表现才会随之更新。

举个 🌰 1

<template><div class="container"><p>count:{{ count }}</p><p>double:{{ doubleCount }}</p><button @click="count++" class="btn">increase</button></div>
</template><script setup>
import { ref } from 'vue'
const count = ref(0)
const doubleCount = ref(count.value * 2)
</script>

解释:

1、render 函数关联了 count 和 doubleCount 数据,count 是响应式数据,因此,count 变化,页面刷新。

2、为什么 doubleCount 没有变化?doubleCount 只是和 count 进行关联,但是响应式本质没有数据与数据的关联,是数据和函数的关联。因此 doubleCount 不会改变。

🌰 2

javascript">import { ref, watchEffect } from 'vue'
const count = ref(0)
const doubleCount = ref(0)
watchEffect(() => {doubleCount.value = count.value * 2
})

解释:

1、现在被监控的函数有两个,render 和 watchEffect,在 watchEffect 中使用了 count,并且 count 是响应式数据,也就是说,render 函数关联两个数据,wacthEffect 关联一个数据。

2、当点击按钮改变 count 时,两个函数都会运行,wacthEffect 运行时,doubleCount 重新赋值,重新展示。

🌰 3

javascript">import { ref, watchEffect } from 'vue'
const count = ref(0)
const useDouble = (count) => {const doubleCount = ref(count)watchEffect(() => {doubleCount.value = count * 2})return doubleCount
}
const doubleCount = useDouble(count.value)

解释:

1、为什么 doubleCount 没有变化?因为 useDouble 函数并没有被监控,只是一个普通的函数。

2、在 useDouble 函数中,有被监控的 wacthEffect 函数,但是函数内部没有读取某个属性,读取的 count 不是响应式数据,而是一个原始数据。

因此,count 变化后,render 运行,但是 watchEffect 不会执行。

🌰 4

javascript">import { computed, ref } from 'vue'
const count = ref(0)
const doubleCount = computed(() => count.value * 2)

解释:

computed 被监控的函数关联了响应式数据 count,因此 count 变化后,render 函数和 computed 函数都会重新运行。

🌰 5

javascript">import { computed, ref } from 'vue'
const count = ref(0)
const useDouble = (count) => {const doubleCount = computed(() => count * 2)return doubleCount
}
const doubleCount = useDouble(count.value)

解释:

useDouble 是普通函数,没有被监控,内部被监控的 computed 没有关联响应式数据,也没有读取某个对象的某个属性。因此 count 变化时,computed 函数不会重新运行。

🌰 6

javascript">import { computed, ref } from 'vue'
const count = ref(0)
const useDouble = (count) => {const doubleCount = computed(() => count.value * 2)return doubleCount
}
const doubleCount = useDouble(count)

解释:

为什么这种写法可以改变 doubleCount 呢?因为 computed 函数读取了对象的属性,也就是我们传递的是响应式对象,而不是一个普通数据。实现了数据和函数直接的关联。

以上就是 Vue 响应式原理的本质,当我们遇到某些疑惑不能解决时,不妨分析一下其根源。


http://www.ppmy.cn/news/1536997.html

相关文章

PLM预训练语言模型Pre-trained Language Model

预训练语言模型&#xff08;Pre-trained Language Model&#xff0c;PLM&#xff09; gpt就是一个典型的例子 一、PLM 的定义与概念 预训练语言模型是一种在大规模文本数据上进行无监督学习得到的语言模型。它通过学习语言的统计规律、语法结构和语义表示&#xff0c;为各种自…

vue-live2d看板娘集成方案设计使用教程

文章目录 前言v1.1.x版本&#xff1a;vue集成看板娘&#xff08;暂不使用&#xff0c;在v1.2.x已替换&#xff09;集成看板娘实现看板娘拖拽效果方案资源备份存储 当前最新调研&#xff1a;2024.10.2开源方案1&#xff1a;OhMyLive2D&#xff08;推荐&#xff09;开源方案2&…

【Webpack--019】TreeShaking

&#x1f913;&#x1f60d;Sam9029的CSDN博客主页:Sam9029的博客_CSDN博客-前端领域博主 &#x1f431;‍&#x1f409;若此文你认为写的不错&#xff0c;不要吝啬你的赞扬&#xff0c;求收藏&#xff0c;求评论&#xff0c;求一个大大的赞&#xff01;&#x1f44d;* &#x…

rtmp协议转websocketflv的去队列积压

websocket server的优点 websocket server的好处&#xff1a;WebSocket 服务器能够实现实时的数据推送&#xff0c;服务器可以主动向客户端发送数据 1 不需要客户端不断轮询。 2 不需要实现httpserver跨域。 在需要修改协议的时候比较灵活&#xff0c;我们发送数据的时候比较…

浅谈C++之Mqtt协议

一、基本介绍 MQTT(Message Queuing Telemetry Transport&#xff0c;消息队列遥测传输协议)&#xff0c;是一种基于发布/订阅模式的"轻量级"通讯协议&#xff0c;它和 Modbus TCP 一样都是基于 TCP/IP 之上的应用层协议。 二、简单用例 在C中实现MQTT通信&#xff…

什么是唯一性约束(UNIQUE Constraint)?

唯一性约束是数据库表设计中的一种完整性约束&#xff0c;用于确保列或一组列中的所有值都是唯一的。 这意味着在一个给定的列或列组内&#xff0c;不能有两个行拥有相同的值。 唯一性约束通常用来维护数据的完整性&#xff0c;并防止重复记录。 在关系型数据库管理系统&…

ctfshow-web 萌新题

给她 spring漏洞 pyload: 1.dirsearch扫描&#xff0c;发现git 2. GitHack工具得到.git文件 <?php $passsprintf("and pass%s",addslashes($_GET[pass])); $sqlsprintf("select * from user where name%s $pass",addslashes($_GET[name])); ?>…

Redis面试篇3

1、Redis的数据类型&#xff0c;以及每种数据类型的使用场景&#xff1f; 常见的几种数据类型和使用场景如下&#xff1a; 字符串(String)&#xff1a;字符串类型是Redis最基本的数据结构&#xff0c;一个键最大能存储512MB。 使用场景&#xff1a;适用于计数器、分布式锁、缓…