Vue: ref、reactive、shallowRef、shallowReactive

1. ref

作用 :创建响应式数据引用(支持基本类型和对象)。
原理

  • 通过 value 属性访问/修改数据

  • 对象类型内部自动转换为 reactive

  • 使用 Object.defineProperty 实现响应式

API

TypeScript 复制代码
import { ref } from 'vue'

const count = ref(0) // 基本类型
const user = ref({ name: 'Alice' }) // 对象类型自动转为 reactive

// 修改值
count.value++ 
user.value.name = 'Bob'

模板中使用

TypeScript 复制代码
<template>
  <div>{{ count }}</div>
  <button @click="count++">Increment</button>
</template>

2. reactive

作用 :创建深层响应式对象(仅支持对象/数组)。
原理

  • 基于 Proxy 实现深度响应式

  • 嵌套对象自动转为 reactive

  • 直接访问属性(无需 .value)

API

TypeScript 复制代码
import { reactive } from 'vue'

const state = reactive({
  count: 0,
  user: { name: 'Alice' } // 嵌套对象自动响应式
})

// 修改值
state.count++
state.user.name = 'Bob'

模板中使用

html 复制代码
<template>
  <div>{{ state.count }}</div>
  <button @click="state.count++">Increment</button>
</template>

3. shallowRef

作用 :创建浅层响应式引用(仅顶层 value 响应)。
原理

  • 只追踪 .value 变化(不追踪内部属性)

  • 对象内部修改 不会 触发更新

  • 替换整个 .value 触发更新

API

TypeScript 复制代码
import { shallowRef } from 'vue'

const user = shallowRef({ name: 'Alice' })

//  ❌ 内部修改无效(不会触发更新)
user.value.name = 'Bob'

// ✅ 替换整个对象有效(触发更新)
user.value = { name: 'Bob' }

典型场景

  • 大型不可变数据(如三方库实例)

  • 性能敏感场景避免深度监听

4. shallowReactive

作用 :创建浅层响应式对象(仅顶层属性响应)。
原理

  • 只响应顶层属性变化

  • 嵌套对象 不会 自动转为响应式

API

TypeScript 复制代码
import { shallowReactive } from 'vue'

const state = shallowReactive({
  count: 0,
  user: { name: 'Alice' } // 嵌套对象非响应式
})

// ✅ 顶层修改有效
state.count++ 

// ❌ 嵌套修改无效(不会触发更新)
state.user.name = 'Bob'

典型场景

  • 表单分组(只关心组变化)

  • 性能优化(避免深度监听大对象)

总结对比表

特性 ref reactive shallowRef shallowReactive
支持类型 所有类型 对象/数组 所有类型 对象/数组
深层响应式 ✅ (对象自动转)
响应触发条件 .value 变化 属性修改 .value 替换 顶层属性修改
访问方式 需要 .value 直接访问 需要 .value 直接访问
性能开销 中等 较高(深度监听)
典型场景 基本类型/对象引用 复杂状态对象 大型不可变数据 层级明确的数据

代码示例总结

TypeScript 复制代码
<script setup>
import { ref, reactive, shallowRef, shallowReactive } from 'vue'

// 1. ref(深度响应)
const deepRef = ref({ count: 0 })
deepRef.value.count++ // 触发更新

// 2. reactive(深度响应)
const deepReactive = reactive({ user: { name: 'Alice' } })
deepReactive.user.name = 'Bob' // 触发更新

// 3. shallowRef(浅层响应)
const shallow = shallowRef({ count: 0 })
shallow.value.count++ //  ❌ 不触发更新
shallow.value = { count: 1 } // ✅ 触发更新

// 4. shallowReactive(浅层响应)
const shallowState = shallowReactive({ 
  user: { name: 'Alice' } 
})
shallowState.user.name = 'Bob' //  ❌ 不触发更新
shallowState.user = { name: 'Bob' } // ✅ 触发更新
</script>

选择建议

  • ref 处理基本类型或需要整体替换的对象

  • reactive 处理深度嵌套的复杂状态

  • shallowXXX 优化大型数据或明确不需要深度监听的场景

相关推荐
yigenhuochai几秒前
Trae Solo 开发体验:从零到完整考试备考平台的奇妙之旅
前端·trae
夏目友人爱吃豆腐3 分钟前
uniapp源码解析(Vue3/Vite版)
前端·vue.js·uni-app
OlahOlah9 分钟前
解决 JavaScript Number 精度问题:处理超大 Long 类型 ID
javascript
JarvanMo10 分钟前
Dart 3.10中的新的lint规则
前端
爱心发电丶12 分钟前
基于UniappX开发电销APP,实现通话录音上传、通时通次
前端
sxjk198718 分钟前
华为IMS系统主要接口备忘
运维·服务器·前端·核心网
T***u33333 分钟前
前端Server Components性能分析 Server Components架构原理
前端
Q***f63536 分钟前
前端动画性能优化,60fps实现技巧
前端
艾莉丝努力练剑38 分钟前
【自动化测试实战篇】Web自动化测试实战:从用例编写到报告生成
前端·人工智能·爬虫·python·pycharm·自动化·测试
Mintopia38 分钟前
💥 Trae Solo 编程 vs. Cursor:新机遇与新挑战
前端·人工智能·trae