简单的聊一聊Vue中如何使用 Ref 和 Reactive 声明响应式数据

在使用Options API时声明响应式数据非常简单。data选项中的所有内容都会自动变为响应式,并在模板中可用。唯一需要注意的是将data设置为一个函数,并防止在所有组件实例之间共享状态。

使用Composition API并不那么简单。状态声明必须明确地使用两个可用的实用函数(ref和reactive)来完成,并且在开始时需要注意多个特殊情况。

Reactive

让我们从简单的开始,Reactive。它接受一个对象或数组,并使用JavaScript代理使其具有响应性。

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

const state = reactive({ count: 0 })

State 变量的行为与普通对象完全相同。

go 复制代码
<script setup>
import { reactive } from 'vue'

const state = reactive({ count: 0 })

function increment() {
  state.count++
}
</script>

<template>
  <button @click="increment">
    {{ state.count }}
  </button>
</template>

同样适用于具有多个层级的对象。

这很简单!但是有个问题。这只适用于复杂类型( Objects 和 Arrays )和集合类型( Maps 和 Sets )。对于原始类型如 string 、 number 或 boolean ,我们需要采用不同的方法。

Ref

为了克服这个限制,我们需要使用 ref 实用工具。声明方式与响应式类似。

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

const count = ref(0)

Ref 接受一个值并返回一个响应式对象。该值在对象内部以 .value 属性的形式返回对应的值。

go 复制代码
const count = ref(0)

console.log(count) // { value: 0 }
console.log(count.value) // 0

count.value++
console.log(count.value) // 1

在模板中使用时,不需要 .value

go 复制代码
<script setup>
import { ref } from 'vue'

const count = ref(0)

function increment() {
  count.value++
}
</script>

<template>
  <button @click="increment">
    {{ count }} // no .value needed
  </button>
</template>

但要小心!这只适用于顶级属性。以下代码片段将产生 [object Object] 。

go 复制代码
// DON'T DO THIS
<script setup>
import { ref } from 'vue'

const object = { foo: ref(1) }

</script>

<template>
  {{ object.foo + 1 }}  // [object Object]
</template>

为了解决这个问题,我们需要将 foo 设置为顶级属性。

go 复制代码
<script setup>
import { ref } from 'vue'

const object = { foo: ref(1) }
const { foo } = object
</script>

<template>
  {{ foo + 1 }}  // This works as expected
</template>

请注意,对于数组和集合类型,也需要使用 .value 。

go 复制代码
const books = reactive([ref('Book title')])

console.log(books[0].value) // need .value here

const map = reactive(new Map([['count', ref(0)]]))

console.log(map.get('count').value) // need .value here

非响应性数据

以前声明非响应式数据并不直观。通常所有内容都声明在data中,这会创建不必要的观察者并导致轻微(可以忽略不计)的性能下降。

go 复制代码
<script setup>
const path = `path/to/image`;
</script>

<template>
  <img :src='path' />
</template>

Readonly(只读)

可以使用 readonly 来防止对状态进行更改。它接受一个对象(响应式或普通)或一个ref,并返回一个不可变的代理。

go 复制代码
const state = reactive({ count: 0 })

const copy = readonly(state)

state.count++ // Works fine

copy.count++ // Fails with a warning

结束

由于文章内容篇幅有限,今天的内容就分享到这里,文章结尾,我想提醒您,文章的创作不易,如果您喜欢我的分享,请别忘了点赞和转发,让更多有需要的人看到。同时,如果您想获取更多前端技术的知识,欢迎关注我,您的支持将是我分享最大的动力。我会持续输出更多内容,敬请期待。

相关推荐
诗书画唱1 分钟前
【前端面试题】JavaScript 核心知识点解析(第二十二题到第六十一题)
开发语言·前端·javascript
excel7 分钟前
前端必备:从能力检测到 UA-CH,浏览器客户端检测的完整指南
前端
前端小巷子14 分钟前
Vue 3全面提速剖析
前端·vue.js·面试
悟空聊架构21 分钟前
我的网站被攻击了,被干掉了 120G 流量,还在持续攻击中...
java·前端·架构
CodeSheep22 分钟前
国内 IT 公司时薪排行榜。
前端·后端·程序员
尖椒土豆sss26 分钟前
踩坑vue项目中使用 iframe 嵌套子系统无法登录,不报错问题!
前端·vue.js
遗悲风27 分钟前
html二次作业
前端·html
江城开朗的豌豆30 分钟前
React输入框优化:如何精准获取用户输入完成后的最终值?
前端·javascript·全栈
CF14年老兵31 分钟前
从卡顿到飞驰:我是如何用WebAssembly引爆React性能的
前端·react.js·trae
画月的亮34 分钟前
前端处理导出PDF。Vue导出pdf
前端·vue.js·pdf