Ref vs. Reactive:Vue 3 响应式变量的最佳选择指南
Ref vs. Reactive:Vue 3 响应式变量的最佳选择指南
在 Vue 3 的 Composition API 中,ref
和 reactive
是创建响应式数据的两种主要方式。许多开发者经常困惑于何时使用哪种方式。本文将深入对比两者的差异,帮助您做出最佳选择。
核心概念解析
1. ref - 值引用容器
import { ref } from 'vue';// 创建 ref
const count = ref(0);// 访问值
console.log(count.value); // 0// 修改值
count.value = 1;
特点:
- 包裹基本类型或对象
- 通过
.value
访问实际值 - 模板中自动解包(无需
.value
)
2. reactive - 对象代理
import { reactive } from 'vue';// 创建 reactive 对象
const state = reactive({count: 0,user: {name: 'John',age: 30}
});// 直接访问属性
console.log(state.count); // 0// 修改嵌套属性
state.user.age = 31;
特点:
- 只接受对象类型
- 深度响应式
- 直接访问属性(无需
.value
)
底层原理差异
内存结构对比:
五大关键维度对比
1. 数据类型支持
类型 | ref 支持 | reactive 支持 |
---|---|---|
字符串 | ✅ | ❌ |
数字 | ✅ | ❌ |
布尔值 | ✅ | ❌ |
数组 | ✅ | ✅ |
对象 | ✅ | ✅ |
Map/Set | ✅ | ✅ |
2. 解构行为对比
ref 的解构问题与解决方案:
const position = {x: ref(0),y: ref(0)
};// 解构后丢失响应性
const { x } = position;// 正确解构方式
const { x } = toRefs(position);
reactive 的解构陷阱:
const state = reactive({ count: 0 });// 解构基本类型属性 - 丢失响应性!
const { count } = state;// 解构对象属性 - 保持响应性
const { user } = state;
3. TypeScript 支持
ref 的类型推导:
const count = ref<number>(0); // Ref<number>// 自动推导
const user = ref({ name: 'John', age: 30 }); // Ref<{ name: string; age: number }>
reactive 的类型限制:
interface State {count: number;user: {name: string;age: number;};
}const state = reactive<State>({count: 0,user: {name: 'John',age: 30}
});
4. 性能考量
创建性能对比(10,000个数据点):
axis | ref | reactive |
---|---|---|
series | 12 | 28 |
更新性能对比:
更新性能对比 | |
---|---|
ref基本类型更新 | 15 |
ref对象属性更新 | 22 |
reactive属性更新 | 18 |
5. 模板使用差异
ref 在模板中:
<template><!-- 自动解包,无需 .value --><div>{{ count }}</div><!-- 对象ref需要.value访问属性 --><div>{{ user.value.name }}</div>
</template><script setup>
const count = ref(0);
const user = ref({ name: 'John' });
</script>
reactive 在模板中:
<template><!-- 直接访问属性 --><div>{{ state.count }}</div><div>{{ state.user.name }}</div>
</template><script setup>
const state = reactive({count: 0,user: { name: 'John' }
});
</script>
最佳实践指南
推荐使用 ref 的场景
-
基本类型值:字符串、数字、布尔值
const isLoading = ref(false); const message = ref('Hello');
-
需要替换整个对象时:
const user = ref({ name: 'John' });// 完全替换对象 user.value = { name: 'Alice' };
-
在组合函数中返回值:
function useCounter() {const count = ref(0);const increment = () => count.value++;return {count,increment}; }
-
模板引用元素:
<template><div ref="container"></div> </template><script setup> const container = ref(null); </script>
推荐使用 reactive 的场景
-
复杂状态对象:
const formState = reactive({username: '',password: '',remember: false });
-
需要深度嵌套的结构:
const appState = reactive({user: {profile: {name: 'John',address: {city: 'New York'}}},settings: { /* ... */ } });
-
与第三方库集成:
const chartData = reactive({labels: ['Jan', 'Feb', 'Mar'],datasets: [{data: [30, 40, 35]}] });// 直接传递给图表库 myChart.update(chartData);
危险操作警示
-
reactive 的直接替换:
let state = reactive({ count: 0 });// 错误!失去响应性 state = { count: 1 };// 正确做法:修改属性 state.count = 1;
-
解构 reactive 基本类型:
const state = reactive({ count: 0 });// 错误!count 失去响应性 const { count } = state;
-
混合使用 ref 和 reactive:
const state = reactive({// 避免!导致双重包装counter: ref(0) });// 访问变得冗长 console.log(state.counter.value);
高级模式与技巧
1. 使用 toRef 保持响应性
const state = reactive({firstName: 'John',lastName: 'Doe'
});// 将 reactive 属性转为 ref
const firstNameRef = toRef(state, 'firstName');// 修改源属性会更新 ref
state.firstName = 'Jane';
console.log(firstNameRef.value); // 'Jane'// 修改 ref 会更新源属性
firstNameRef.value = 'Alice';
console.log(state.firstName); // 'Alice'
2. toRefs 的强大转换
function useFeature() {const state = reactive({x: 0,y: 0});// 将 reactive 对象转换为 ref 集合return toRefs(state);
}// 在组件中使用
const { x, y } = useFeature();
3. 使用 shallowRef 优化性能
const largeObject = shallowRef({ /* 巨大对象 */ });// 修改内部属性不会触发更新
largeObject.value.nested.prop = 'new value';// 需要强制触发更新
largeObject.value = { ...largeObject.value };
性能优化指南
1. 大型数据集处理
2. 更新策略对比
方法 | 适用场景 | 性能影响 |
---|---|---|
ref.value = | 基本类型更新 | ⭐⭐ |
reactive.prop = | 对象属性更新 | ⭐⭐ |
Object.assign | 多个属性更新 | ⭐⭐⭐ |
不可变数据 | 复杂状态更新 | ⭐⭐⭐⭐ |
决策流程图
总结与实践建议
-
黄金法则:
- 基本类型 → ref
- 对象/数组 → reactive
- 需要替换整个对象 → ref
- 需要解构属性 → reactive + toRefs
-
性能关键点:
- 大型对象使用
shallowRef
- 避免在循环中创建深层 reactive
- 批量更新使用
nextTick
- 大型对象使用
-
组合函数最佳实践:
// 推荐:返回 ref 或 toRefs 转换 export function useFeature() {const state = reactive({ x: 0, y: 0 });return toRefs(state); }// 或返回 ref 集合 export function useCounter() {const count = ref(0);return { count }; }
-
终极建议:
“在 Vue 3 项目中,我推荐使用 ref 作为默认选择,仅在处理深度嵌套的对象结构时使用 reactive。这种策略简化了代码一致性,减少了 .value 使用的认知负荷,同时保持了灵活性。”
通过理解这些核心差异和应用场景,您将能够更自信地在 Vue 3 项目中选择正确的响应式 API,编写出更高效、更可维护的代码。
相关文章:
Ref vs. Reactive:Vue 3 响应式变量的最佳选择指南
Ref vs. Reactive:Vue 3 响应式变量的最佳选择指南 在 Vue 3 的 Composition API 中,ref 和 reactive 是创建响应式数据的两种主要方式。许多开发者经常困惑于何时使用哪种方式。本文将深入对比两者的差异,帮助您做出最佳选择。 核心概念解…...
让视觉基础模型(VFMs)像大语言模型(LLMs)一样“会思考”
视觉检测器的演进:从 DETR 到 Grounding-DINO DINO-R1 的基础是 Grounding-DINO,而 Grounding-DINO 本身是一系列视觉检测器演进的结果。理解这个发展过程对掌握 DINO-R1 的核心技术至关重要。 DETR:用 Transformer 革新目标检测 在 DETR&…...
现代前端框架的发展与演进
现代前端框架的发展与演进是一个非常值得关注的话题,反映了整个前端生态系统的不断演化与技术深度的提升。以下是这一趋势的详细解析: 📈 现代前端框架的发展与演进 🔹 第一阶段:jQuery 时代(2006-2013&am…...

【LLM-Agent】智能体的记忆缓存设计
note 实践:https://modelscope-agent.readthedocs.io/zh-cn/latest/modules/memory.html 文章目录 note一、Agent的记忆实现二、相关综述三、记忆体的构建四、cursor的记忆设计1. 记忆生成提示词2. 记忆评估提示词 五、记忆相关的MCPReference 一、Agent的记忆实现…...

一起学Spring AI:核心概念
人工智能概念 本节描述了 Spring AI 使用的核心概念。我们建议您仔细阅读,以理解 Spring AI 实现背后的思想。 模型(Models) 人工智能模型是设计用来处理和生成信息的算法,通常模仿人类的认知功能。通过从大型数据集中学习模式…...
Oracle业务用户的存储过程个数及行数统计
Oracle业务用户的存储过程个数及行数统计 统计所有业务用户存储过程的个数独立定义的存储过程定义在包里的存储过程统计所有业务用户存储过程的总行数独立定义的存储过程定义在包里的存储过程📖 对存储过程进行统计主要用到以下三个系统视图: dba_objects:记录了所有独立创…...

PicSharp(图片压缩工具) v1.1.6
PicSharp 一个简单、高效、灵活的跨平台桌面图像压缩应用程序。软件基于Rust实现,高性能低资源,能快速扫描文件或目录,批处理图像。软件还具备组合压缩策略,TinyPNG提供最佳压缩比,但需要互联网连接,对大量…...

前端文件下载常用方式详解
在前端开发中,实现文件下载是常见的需求。根据不同的场景,我们可以选择不同的方法来实现文件流的下载。本文介绍三种常用的文件下载方式: 使用 axios 发送 JSON 请求下载文件流使用 axios 发送 FormData 请求下载文件流使用原生 form 表单提…...

【DAY42】Grad-CAM与Hook函数
内容来自浙大疏锦行python打卡训练营 浙大疏锦行 知识点: 回调函数lambda函数hook函数的模块钩子和张量钩子Grad-CAM的示例 作业:理解下今天的代码即可 在深度学习中,我们经常需要查看或修改模型中间层的输出或梯度。然而,标准的前向传播和反…...

如何生成和制作PDF文件
在数字化办公的今天,PDF文件已经成为我们工作和学习中不可或缺的一部分。无论是合同、报告、简历,还是电子书、表单,PDF格式都以其跨平台兼容性、不可编辑性和清晰的排版而被广泛使用。但你是否知道,生成和制作PDF文件其实并不复杂…...

【K8S系列】Kubernetes 中 Pod(Java服务)启动缓慢的深度分析与解决方案
本文针对 Kubernetes 中 Java 服务启动时间慢的深度分析与解决方案文章,结合了底层原理、常见原因及具体优化策略: Kubernetes 中 Java 服务启动缓慢的深度分析与高效解决方案 在 Kubernetes 上部署 Java 应用时,启动时间过长是常见痛点,尤其在需要快速扩缩容或滚动更新的…...

【Java学习笔记】StringBuilder类(重点)
StringBuilder(重点) 1. 基本介绍 是一个可变的字符串序列。该类提供一个与 StringBuffer 兼容的 API,但不保证同步(StringBuilder 不是线程安全的) 该类被设计用作 StringBuffer 的一个简易替换,用在字符…...
JavaScript ES6 解构:优雅提取数据的艺术
JavaScript ES6 解构:优雅提取数据的艺术 在 JavaScript 的世界中,ES6(ECMAScript 2015)的推出为开发者带来了许多革命性的特性,其中“解构赋值”(Destructuring Assignment)无疑是最受欢迎的功…...

iview Switch Tabs TabPane 使用提示Maximum call stack size exceeded堆栈溢出
在vue项目中使用iview 框架部分组件时,直接引入使用报Maximum call stack size exceeded image.png 堆栈溢出 解决方案 更换组件名称就可以了 image.png 或 image.png 就可以了 猜测是因为和vue自己提供的组件名称一致了,重名问题导致的,具体…...

基于Halcon深度学习之分类
***** ***环境准备*** ***系统:win7以上系统 ***显卡:算力3.0以上 ***显卡驱动:10.1以上版本(nvidia-smi查看指令)***读取深度学习模型*** read_dl_model (pretrained_dl_classifier_compact.hdl, DLModelHandle) ***获…...
零基础在实践中学习网络安全-皮卡丘靶场(第十五期-URL重定向模块)
本期内容和之前的CSRF,File inclusion有联系,复习后可以更好了解 介绍 不安全的url跳转 不安全的url跳转问题可能发生在一切执行了url地址跳转的地方。如果后端采用了前端传进来的(可能是用户传参,或者之前预埋在前端页面的url地址)参数作为了跳转的目…...

技巧小结:根据寄存器手册写常用外设的驱动程序
需求:根据STM32F103寄存器手册写DMA模块的驱动程序 一、分析标准库函数的写法: 各个外设的寄存器地址定义在stm32f10x.h文件中:此文件由芯片厂家提供;内核的有关定义则定义在core_cm3.h文件中:ARM提供; 1、查看外设区域多级划分…...

设计模式(代理设计模式)
代理模式解释清楚,所以如果想对一个类进行功能上增强而又不改变原来的代码情况下,那么只需要让这个类代理类就是我们的顺丰,对吧?并行增强就可以了。具体增强什么?在哪方面增强由代理类进行决定。 代码实现就是使用代理对象代理相关的逻辑…...

从代码学习深度强化学习 - 初探强化学习 PyTorch版
文章目录 前言强化学习的概念强化学习的环境强化学习中的数据强化学习的独特性总结前言 本文将带你初步了解强化学习 (Reinforcement Learning, RL) 的基本概念,并通过 PyTorch 实现一些简单的强化学习算法。强化学习是一种让智能体 (agent) 通过与环境 (environment) 的交互…...
AI大神吴恩达-提示词课程笔记
如何有效编写提示词 在学习如何与语言模型(如ChatGPT)交互时,编写清晰且高效的提示词(Prompt)是至关重要的。本课程由ESA提供,重点介绍了提示词工程(Prompt Engineering)的两个核心…...
ArcGIS Pro 3.4 二次开发 - 地图探索
环境:ArcGIS Pro SDK 3.4 + .NET 8 文章目录 地图探索1 地图视图1.1 测试视图是否为3D1.2 设置视图模式1.3 启用视图链接2 更新地图视图范围2.1 返回上一个相机视图2.2 切换到下一个相机视角2.3 缩放到全图范围2.4 固定放大2.5 固定缩小2.6 缩放到范围2.7 缩放到一个点2.8 缩放…...

ELK日志管理框架介绍
在小铃铛的毕业设计中涉及到了ELK日志管理框架,在调研期间发现在中文中没有很好的对ELK框架进行介绍的文章,因此拟在本文中进行较为详细的实现的介绍。 理论知识 ELK 框架介绍 ELK 是一个流行的开源日志管理解决方案堆栈,由三个核心组件组…...

【Linux】sed 命令详解及使用样例:流式文本编辑器
【Linux】sed 命令详解及使用样例:流式文本编辑器 引言 sed 是 Linux/Unix 系统中一个强大的流式文本编辑器,名称来源于 “Stream EDitor”(流编辑器)。它允许用户在不打开文件的情况下对文本进行筛选和转换,是命令行…...

机器学习:聚类算法及实战案例
本文目录: 一、聚类算法介绍二、分类(一)根据聚类颗粒度分类(二)根据实现方法分类 三、聚类流程四、K值的确定—肘部法(一)SSE-误差平方和(二)肘部法确定 K 值 五、代码重…...
预览pdf(url格式和blob格式)
<template><div class"pdf-container"><div v-if"loading" class"loading-state"><a-spin size"large" /></div><div v-else-if"error" class"loading-state">加载失败&…...

【p2p、分布式,区块链笔记 MESH】 论文阅读 Thread/OpenThread Low-Power Wireless Multihop Net
paperauthorThread/OpenThread: A Compromise in Low-Power Wireless Multihop Network Architecture for the Internet of ThingsHyung-Sin Kim, Sam Kumar, and David E. Culler 目录 引言RPL 标准设计目标与架构设计选择与特性shortcomIngs of RPL设计选择的反面影响sImulta…...
for AC500 PLCs 3ADR025003M9903的安全说明
1安全说明 必须遵守特殊的环境条件(例如,由于爆炸性物质、重污染或腐蚀影响的危险区域)。必须在指定的技术数据和系统数据范围内处理和操作设备。该装置不含可维修部件,不得打开。除非另有规定,否则操作过程中必须关闭可拆卸的盖子。拒绝对不…...

moon游戏服务器-demo运行
下载地址 https://github.com/sniper00/MoonDemo redis安装 Redis-x64-3.0.504.msi 服务器配置文件 D:\gitee\moon_server_demo\serverconf.lua 貌似不修改也可以的,redis不要设置密码 windows编译 安装VS2022 Community 下载premake5.exe放MoonDemo\server\moon 双…...
前端(vue)学习笔记(CLASS 7):vuex
vuex概述 vuex是一个vue的状态管理工具,状态就是数据 大白话:vuex是一个插件,可以帮我们管理vue通用的数据(多组件共享的数据) 场景 1、某个状态在很多个组件来使用(个人信息) 2、多个组件…...
[特殊字符] 在 React Native 项目中封装 App Icon 一键设置命令(支持参数与默认路径)
📦 前置依赖 使用的是社区维护的 CLI 工具: @bam.tech/react-native-make它扩展了 react-native 命令,支持 set-icon 功能。 安装: yarn add -D "@bam.tech/react-native-make"🧠 封装目标 我们希望能够通过以下方式调用: # 默认使用 ./icon.png yarn …...