在 Vue3 中,有许多与响应式相关的函数,例如 toRef、toRefs、isRef、unref 等等。合理地使用这些函数可以在实际开发中大大提高效率。本文将详细介绍这些函数的用法,让我们在实际开发中知道应该使用哪些 API 并能够熟练地回答面试官的相关问题。

ref()

大家对于 ref 这个 API 肯定都不陌生。在 Vue3 中经常会用到它。它的作用是接收一个值并返回一个响应式的对象。我们可以通过.value 属性来访问和修改这个值。在模板中,我们可以省略.value,例如在下面的代码中,当点击按钮时,页面中的 count 会响应式地更改。

  1. <template>
  2. <div>
  3. {{ count }}
  4. <button @click="addCount">+1</button>
  5. </div>
  6. </template>
  7. <script lang='ts' setup>
  8. import { ref } from "vue"
  9. const count = ref(1)
  10. const addCount = () => {
  11. count.value++
  12. }
  13. </script>

toRef

toRef 可以根据一个响应式对象中的一个属性,创建一个响应式的 ref。同时这个 ref 和原对象中的属性保持同步,改变原对象属性的值这个 ref 会跟着改变,反之改变这个 ref 的值原对象属性值也会改变,它接收两个参数,一个是响应式对应,另一个则是属性值,例如下面代码

  1. <template>
  2. <div>
  3. {{ count.a }}
  4. {{ a }}
  5. <button @click="addCount">+1</button>
  6. </div>
  7. </template>
  8. <script lang='ts' setup>
  9. import { ref, toRef } from "vue"
  10. const count = ref({
  11. a: 1,
  12. b: 2
  13. })
  14. const a = toRef(count.value, 'a')
  15. const addCount = () => {
  16. a.value++
  17. }
  18. </script>

点击按钮的时候修改了 a 的值,此时 count 中的 a 也会跟着修改,当然这里的 count 也可以用 reactive

toRefs

toRefs 它可以将一个响应式对象转成普通对象,而这个普通对象的每个属性都是响应式的 ref

  1. <template>
  2. <div>
  3. {{ count.a }}
  4. {{ countAsRefs.a }}
  5. <button @click="addCount">+1</button>
  6. </div>
  7. </template>
  8. <script lang='ts' setup>
  9. import { reactive, toRefs } from "vue"
  10. const count = reactive({
  11. a: 1,
  12. b: 2
  13. })
  14. const countAsRefs = toRefs(count)
  15. const addCount = () => {
  16. countAsRefs.a.value++
  17. }
  18. </script>

此时代码中的countAsRefs类型为

  1. {
  2. a: Ref<number>,
  3. b: Ref<number>
  4. }

它的属性 a 和 b 都是响应式的 ref 对象,同样的它们和原对象的 count 的属性也是保持同步的

根据它的特性我们通常用它来解构一个响应式对象而不会让其失去响应式

  1. import { reactive, toRefs } from "vue";
  2. const count = reactive({
  3. a: 1,
  4. b: 2,
  5. });
  6. const { a, b } = toRefs(count);

此时的 a 和 b 都是一个响应式的 ref 对象,并和原对象的 a 和 b 属性保持同步

isRef()

isRef 顾名思义它是用来判断某个值是否是 ref,注意:它判断不了这个值是不是 reactive(可以使用 isReactive 判断)

  1. import { reactive, isRef, ref } from "vue";
  2. const count = ref(1);
  3. const testObj = reactive({
  4. a: 1,
  5. });
  6. console.log(isRef(count)); //true
  7. console.log(isRef(testObj)); //false

unref()

其实它是一个语法糖

  1. val = isRef(val) ? val.value : val;

如果是 ref 则返回它的内部值,否则则返回它本身。通过这个语法糖我们可以看出它可以对响应式对象解除响应式引用,比如我们只想获取一个响应式的值,但不想要它的响应式可以使用它解除引用。 例如

  1. <template>
  2. <div>
  3. {{ unRefAsCount }}
  4. {{ count }}
  5. <button @click="addCount">+1</button>
  6. </div>
  7. </template>
  8. <script lang='ts' setup>
  9. import { unref, ref } from "vue"
  10. const count = ref(1)
  11. let unRefAsCount = unref(count)
  12. const addCount = () => {
  13. count.value++
  14. }
  15. </script>

代码中的 unRefAsCount 是不具备响应式的

shallowRef

通过翻译我们可以看出它是浅层的 ref,什么是浅层的 ref 呢? 与 ref 不同的是只有.value 是响应式的,再深层的属性则不具备响应式

  1. <template>
  2. <div>
  3. {{ shallowObj.a }}
  4. <button @click="addCount"> +1</button>
  5. </div>
  6. </template>
  7. <script lang='ts' setup>
  8. import { shallowRef } from "vue"
  9. const shallowObj = shallowRef({
  10. a: 1
  11. })
  12. const addCount = () => {
  13. //不会触发页面更新
  14. shallowObj.value.a++
  15. }
  16. </script>

但是如果我们将 addCount 改为修改整个.value 就会触发响应式了

  1. const addCount = () => {
  2. let temp = shallowObj.value.a;
  3. temp++;
  4. shallowObj.value = {
  5. a: temp,
  6. };
  7. };

triggerRef

它可以让浅层的 ref 即 shallowRef 深层属性发生改变的时候强制触发更改,比如上面触发不了响应式的代码示例加入triggerRef

  1. <template>
  2. <div>
  3. {{ shallowObj.a }}
  4. <button @click="addCount"> +1</button>
  5. </div>
  6. </template>
  7. <script lang='ts' setup>
  8. import { shallowRef, triggerRef } from "vue"
  9. const shallowObj = shallowRef({
  10. a: 1
  11. })
  12. const addCount = () => {
  13. shallowObj.value.a++
  14. //加入triggerRef强制触发更改
  15. triggerRef(shallowObj)
  16. }
  17. </script>

此时再看页面效果则触发了响应式

customRef

顾名思义它是自定义的 ref,我们可以通过 customRef 来显式的追踪某个值的响应式变化,它接收一个函数,这个函数接受 track 和 trigger 两个函数作为参数,并返回一个带有 get 和 set 方法的对象。比如下面封装一个自定义的响应式对象 myRef,同时控制它只有值小于 4 才会触发响应式

  1. <template>
  2. <div>
  3. {{ count }}
  4. <button @click="addCount"> +1</button>
  5. </div>
  6. </template>
  7. <script lang='ts' setup>
  8. import { customRef } from "vue"
  9. const myRef = (value: number) => {
  10. const customValue = customRef((track, trigger) => {
  11. return {
  12. get() {
  13. //通知vue需要追踪后续内容的变化,这里可以自由控制
  14. track()
  15. return value
  16. },
  17. set(newValue) {
  18. console.log(newValue);//myRef.value=xxx的xxx值
  19. //加trigger则触发响应式,通知vue更新页面,这里可以自由控制是否加trigger
  20. if(value<4) trigger()
  21. value = newValue
  22. }
  23. }
  24. })
  25. return customValue
  26. }
  27. const count = myRef(0)
  28. const addCount = () => {
  29. count.value++
  30. }
  31. </script>

可以看到当 count 大于 4 的时候便失去了响应式

总结

本篇文章详细介绍了 Vue3 中各种 ref 的用法,其中包括

  • ref(): 接收一个值并返回一个响应式的对象,可以使用.value 属性来访问和修改这个值。
  • toRef(obj, key): 根据一个响应式对象中的一个属性,创建一个响应式的 ref,并且该 ref 和原对象中的属性保持同步。

    -toRefs(obj): 将一个响应式对象转换成一个普通对象,其中普通对象的每个属性都是响应式的 ref。
  • isRef(value): 判断某个值是否是 ref 对象。
  • unref(value): 用于解除响应式引用
  • shallowRef(value): 创建一个浅层的 ref,只有 value 属性是响应式的,深层的属性不具备响应式。
  • triggerRef(ref): 强制浅层的 ref 发生改变时触发响应式。
  • customRef(factory): 自定义 ref 对象,可以显式地追踪某个值的响应式变化。

如果本篇文章对你有所帮助的话那就点个赞吧~

搞懂 Vue3 中的各种 ref:toRef,toRefs,isRef,unref...的更多相关文章

  1. 轻松搞懂Java中的自旋锁

    前言 在之前的文章<一文彻底搞懂面试中常问的各种“锁”>中介绍了Java中的各种“锁”,可能对于不是很了解这些概念的同学来说会觉得有点绕,所以我决定拆分出来,逐步详细的介绍一下这些锁的来龙 ...

  2. 来一轮带注释的demo,彻底搞懂javascript中的replace函数

    javascript这门语言一直就像一位带着面纱的美女,总是看不清,摸不透,一直专注服务器端,也从来没有特别重视过,直到最近几年,javascript越来越重要,越来越通用.最近和前端走的比较近,借此 ...

  3. 帮你彻底搞懂JS中的prototype、__proto__与constructor(图解)

    作为一名前端工程师,必须搞懂JS中的prototype.__proto__与constructor属性,相信很多初学者对这些属性存在许多困惑,容易把它们混淆,本文旨在帮助大家理清它们之间的关系并彻底搞 ...

  4. 让你彻底搞懂JS中复杂运算符==

    让你彻底搞懂JS中复杂运算符== 大家知道,==是JavaScript中比较复杂的一个运算符.它的运算规则奇怪,容易让人犯错,从而成为JavaScript中“最糟糕的特性”之一. 在仔细阅读了ECMA ...

  5. 彻底搞懂 JS 中 this 机制

    彻底搞懂 JS 中 this 机制 摘要:本文属于原创,欢迎转载,转载请保留出处:https://github.com/jasonGeng88/blog 目录 this 是什么 this 的四种绑定规 ...

  6. 一文搞懂 js 中的各种 for 循环的不同之处

    一文搞懂 js 中的各种 for 循环的不同之处 See the Pen for...in vs for...of by xgqfrms (@xgqfrms) on CodePen. for &quo ...

  7. 彻底搞懂JavaScript中的继承

    你应该知道,JavaScript是一门基于原型链的语言,而我们今天的主题 -- "继承"就和"原型链"这一概念息息相关.甚至可以说,所谓的"原型链&q ...

  8. 一文彻底搞懂Java中的环境变量

    一文搞懂Java环境变量 记得刚接触Java,第一件事就是配环境变量,作为一个初学者,只知道环境变量怎样配,在加上各种IDE使我们能方便的开发,而忽略了其本质的东西,只知其然不知其所以然,随着不断的深 ...

  9. 一文带你搞懂java中的变量的定义是什么意思

    前言 在之前的文章中,壹哥给大家讲解了Java的第一个案例HelloWorld,并详细给大家介绍了Java的标识符,而且现在我们也已经知道该使用什么样的工具进行Java开发.那么接下来,壹哥会集中精力 ...

  10. 大白话讲解Promise(三)搞懂jquery中的Promise

    前两篇我们讲了ES6中的Promise以及Promise/A+规范,在Promise的知识体系中,jquery当然是必不可少的一环,所以本篇就来讲讲jquery中的Promise,也就是我们所知道的D ...

随机推荐

  1. 都说DevOps落地难,到底难在哪里?也许你还没找到套路

    当你打开这篇文章的时候,也许你也在为DevOps的落地而苦恼,也许你的组织正在尝试DevOps转型,作为一线的实践者,说说我对这个"落地难"的看法,欢迎交流不同看法- DevOps ...

  2. Django-账户用户忘记密码

    方法1:Terminal命令 python manage.py changepassword admin Password: PY666666 Password (again): PY666666 方 ...

  3. from . import XXX

    [Python]from . import XXX 一. 官方文档 sound/ __init__.py  formats/ __init__.py wavread.py wavwrite.py ai ...

  4. ESlint配置详解

    开发中出现eslint提示代码格式错误,有时候不明白其配置规范,是件很头疼的事情到处找api又是半天:so记录一份配置详情便于开发中翻阅 { // 环境定义了预定义的全局变量. "env&q ...

  5. Python基础 - 算数运算符

    算数运算符   运算符 描述 实例 + 加 - 两个对象相加 a + b 输出结果 30 - 减 - 得到负数或是一个数减去另一个数 a - b 输出结果 -10 * 乘 - 两个数相乘或是返回一个被 ...

  6. docker升级gitlab

    昨天在家部署了gitlab,版本居然是15.10,公司版本却是14.6,升级一波. 官方文档: https://docs.gitlab.com/ee/update/#upgrading-without ...

  7. 看看Angular有啥新玩法!手把手教你在Angular15中集成报表插件

    摘要:本文由葡萄城技术团队于博客园原创并首发.葡萄城为开发者提供专业的开发工具.解决方案和服务,赋能开发者. Angular15新特性 Angular框架(以下简称"Angular" ...

  8. WFP必须掌握的技能之自定义控件——实战:自制上传文件显示进度按钮

    自定义控件在WPF开发中是很常见的,有时候某些控件需要契合业务或者美化统一样式,这时候就需要对控件做出一些改造. 目录 按钮设置圆角 按钮上传文件相关定义 测试代码 话不多说直接看效果 默认效果: 上 ...

  9. 【网络知识】FTP主被动模式介绍及抓包分析

    一.FTP是什么 FTP,即文件传输协议(File Transfer Protocol,FTP),基于该协议客户端与服务端可以实现共享文件.上传文件.下载文件. FTP 基于TCP协议生成两个连接,一 ...

  10. 自然语言处理 Paddle NLP - 任务式对话系统-理论

    什么是任务型对话: 任务型:用于帮助用户完成某领域的特定任务,例如订餐.查天气.订票等 闲聊型:也称作开放域对话系统,目标是让用户持续的参与到交互过程,提供情感陪伴 问答型:提供知识满足,具体类型比较 ...