敲黑板!vue3重点!一文了解Composition API新特性:ref、toRef、toRefs

简介: 该文章深入探讨了Vue3中Composition API的关键特性,包括`ref`、`toRef`、`toRefs`的使用方法与场景,以及它们如何帮助开发者更好地管理组件状态和促进逻辑复用。

Composition API新特性

上一篇文章中,我们初步了解了vue3的新特性,今天,我们将延续 Composition API的话题,来了解 Composition API带来的新特性: reftoReftoRefs

下面开始进入本文的讲解✨

一、🙎如何理解ref、toRef和toRefs

1、ref、toRef和toRefs是什么

(1)ref

1)ref是什么
  • ref 可以生成 值类型(即基本数据类型) 的响应式数据;
  • ref 可以用于模板reative
  • ref 通过 .value 来修改值(一定要记得加上 .value );
  • ref 不仅可以用于响应式,还可以用于模板的 DOM 元素。
2)举个例子🌰

假设我们定义了两个值类型的数据,并通过一个定时器来看它响应式前后的效果。接下来我们用代码来演示一下:

<template>
    <p>ref demo {
  {ageRef}} {
  {state.name}}</p>
</template>

<script>
import {
     ref, reactive } from 'vue'

export default {
    
    name: 'Ref',
    setup(){
    
        const ageRef = ref(18)
        const nameRef = ref('monday')

        const state = reactive({
    
            name: nameRef
        })

        setTimeout(() => {
    
            console.log('ageRef', ageRef.value,'nameRef', nameRef.value)
            ageRef.value = 20
            nameRef.value = 'mondaylab'
            console.log('ageRef', ageRef.value,'nameRef', nameRef.value)
        },1500)

        return{
    
            ageRef,
            state
        }
    }
}
</script>

别眨眼,来看下此时浏览器的显示效果:

ref

大家可以看到,控制台先后打印的顺序是响应式前的数据响应式后的数据。因此,通过 ref ,可以实现值类型的数据响应式


值得注意的是, ref 不仅可以实现响应式,还可以用于模板的DOM元素我们用一段代码来演示一下:

<template>
    <p ref="elemRef">今天是周一</p>
</template>

<script>
import {
     ref, onMounted } from 'vue'

export default {
    
    name: 'RefTemplate',
    setup(){
    
        const elemRef = ref(null)

        onMounted(() => {
    
            console.log('ref template', elemRef.value.innerHTML, elemRef.value)
        })

        return{
    
            elemRef
        }
    }
}
</script>

此时浏览器的显示效果如下所示:

ref

我们通过在模板中绑定一个 ref ,之后在生命周期中调用,最后浏览器显示出该 DOM 元素。所以说, ref 也可以用来渲染模板中的DOM元素

(2)toRef是什么

1)toRef是什么
  • toRef 可以响应对象 Object ,其针对的是某一个响应式对象( reactive 封装)的属性prop

  • toRef 和对象 Object 两者保持引用关系,即一个改完另外一个也跟着改。

  • toRef 如果用于普通对象(非响应式对象),产出的结果不具备响应式。如下代码所示:

//普通对象
const state = {
   
    age: 20,
    name: 'monday'
}
//响应式对象
const state = reactive({
   
    age: 20,
    name: 'monday'
})
2)举个例子🌰

对于一个普通对象来说,如果这个普通对象要实现响应式,就用 reactive 。用了 reactive 之后,它就在响应式对象里面。那么在 一个响应式对象里面,如果其中有一个属性要拿出来单独做响应式的话,就用 toRef 。来举个例子看一看:

<template>
    <p>toRef demo - {
  {ageRef}} - {
  {state.name}} {
  {state.age}}</p>
</template>

<script>
import {
     ref, toRef, reactive, computed } from 'vue'

export default {
    
    name: 'ToRef',
    setup() {
    
        const state = reactive({
    
            age: 18,
            name: 'monday'
        })

        // // toRef 如果用于普通对象(非响应式对象),产出的结果不具备响应式
        // const state = {
    
        //     age: 18,
        //     name: 'monday'
        // }

        //实现某一个属性的数据响应式
        const ageRef = toRef(state, 'age')

        setTimeout(() => {
    
            state.age = 20
        }, 1500)

        setTimeout(() => {
    
            ageRef.value = 25 // .value 修改值
        }, 3000)

        return {
    
            state,
            ageRef
        }
    }
}
</script>

此时我们来看下浏览器的显示效果:

toRef

我们通过 reactive 来创建一个响应式对象,之后呢,如果只单独要对响应式对象里面的某一个属性进行响应式,那么使用toRef 来解决。用 toRef(Object, prop) 的形式来传对象名具体的属性名,达到某个属性数据响应式的效果。

(3)toRefs是什么

1)toRefs是什么
  • toRef 不一样的是, toRefs 是针对整个对象的所有属性,目标在于将响应式对象( reactive 封装)转换为普通对象
  • 普通对象里的每一个属性 prop 都对应一个 ref
  • toRefs 和对象 Object 两者保持引用关系,即一个改完另外一个也跟着改。
2)举个例子🌰

假设我们要将一个响应式对象里面的所有元素取出来,那么我们可以这么处理。代码如下:

<template>
    <p>toRefs demo {
  {state.age}} {
  {state.name}}</p>
</template>

<script>
import {
     ref, toRef, toRefs, reactive } from 'vue'

export default {
    
    name: 'ToRefs',
    setup() {
    
        const state = reactive({
    
            age: 20,
            name: 'monday'
        })

        return {
    
            state
        }
    }
}
</script>

此时浏览器的显示结果如下:

toRefs

但是这样子好像有点略显麻烦,因为在模板编译的时候一直要 state. ,这样如果遇到要取很多个属性的时候就有点臃肿了。


既然太臃肿了,那我们换一种思路,把 state 进行解构。代码如下:

<template>
    <p>toRefs demo {
  {age}} {
  {name}}</p>
</template>

<script>
import {
     ref, toRef, toRefs, reactive } from 'vue'

export default {
    
    name: 'ToRefs',
    setup() {
    
        const state = reactive({
    
            age: 20,
            name: 'monday'
        })

        return {
    
            ...state
        }
    }
}
</script>

此时浏览器的显示结果如下:

toRefs

效果是一样的,看起来清晰了很多。但是呢……天上总不会有无缘无故的馅饼出现,得到一些好处的同时总要失去些原本拥有的东西。


对于解构后的对象来说,如果直接解构 reactive ,那么解构出来的对象会直接失去响应式。我们用一个定时器来检验下效果,具体代码如下:

<template>
    <p>toRefs demo {
  {age}} {
  {name}}</p>
</template>

<script>
import {
     ref, toRef, toRefs, reactive } from 'vue'

export default {
    
    name: 'ToRefs',
    setup() {
    
        const state = reactive({
    
            age: 20,
            name: 'monday'
        })

         setTimeout(() => {
    
             state.age = 25
         }, 1500)

        return {
    
            ...state
        }
    }
}
</script>

此时浏览器的显示结果如下:

toRefs

我们等了好几秒之后,发现 age 迟迟不变成25,所以当我们解构 reactive 的对象时,响应式将会直接失去。


所以,就来到了我们的 toRefstoRefs 在把响应式对象转变为普通对象后,不会丢失掉响应式的功能。具体我们用代码来演示一下:

<template>
    <p>toRefs demo {
  {age}} {
  {name}}</p>
</template>

<script>
import {
     ref, toRef, toRefs, reactive } from 'vue'

export default {
    
    name: 'ToRefs',
    setup() {
    
        const state = reactive({
    
            age: 18,
            name: 'monday'
        })

        const stateAsRefs = toRefs(state) // 将响应式对象,变成普通对象

        setTimeout(() => {
    
            console.log('age', state.age, 'name', state.name)
            state.age = 20,
            state.name = '周一'
            console.log('age', state.age, 'name', state.name)
        }, 1500)

        return stateAsRefs
    }
}
</script>

此时我们观察浏览器的显示效果:

toRefs

大家可以看到,用了 toRefs ,普通对象的值成功被取出来了,并且还不会丢失响应式的功能,该改变的值一个也不少。

(4)合成函数返回响应式对象

了解了上面三种类型的使用,我们再来看一种场景:合成函数如何返回响应式对象下面附上代码:

function useFeatureX(){
       
    const state = reactive({
           
    x: 1,        
    y: 2    
    })        
    //逻辑运行状态,……        
    //返回时转换为ref    
    return toRefs(state)
}
export default{
       setup(){
           
    //可以在不失去响应性的情况下破坏结构        
    const {
   x, y} = useFeatureX()                
    return{
               
        x,            
        y        
    }    
}}

在第一段代码中,我们定义了一个函数,并且用 toRefsstate 对象进行返回,之后在组件里面直接调用响应式对象

通过这样方式,让代码逻辑变得更加清晰明了,复用性更强。

2、最佳使用方式

通过上面的演示可以得出以下几点结论:

  • reactive 做对象的响应式,用 ref值类型的响应式。
  • setup 中返回 toRefs(state) ,或者 toRef(state, 'xxx')
  • 为了防止误会产生, ref 的变量命名尽量都用 xxxRef ,这样在使用的时候会更清楚明了。
  • 合成函数返回响应式对象时,使用 toRefs

3、深入理解

讲完 reftoReftoRefs ,我们再来思考一个问题:为什么一定要用它们呢?可以不用吗?

(1)为什么需要用ref

  • 值类型(即基本数据类型)无处不在,如果不用 ref 而直接返回值类型,会丢失响应式
  • 比如在 setupcomputed合成函数等各种场景中,都有可能返回值类型
  • Vue 如果不定义 ref ,用户将自己制造 ref ,这样反而会更加混乱。

(2)为何ref需要.value属性

通过上面的分析我们知道, ref 需要通过 .value 来修改值。这看起来是一个很麻烦的操作,总是频繁的 .value 感觉特别琐碎。那为什么一定要 .value 呢?我们来揭开它的面纱。

  • ref 是一个对象,这个对象不丢失响应式,且这个对象用 value 来存储值。
  • 因此,通过 .value 属性的 getset 来实现响应式。
  • 只有当用于 模板reactive 时,不需要 .value 来实现响应式,而其他情况则都需要

(3)为什么需要toRef和toRefs

ref 不一样的是, toReftoRefs 这两个兄弟,它们不创造响应式,而是延续响应式。创造响应式一般由 ref 或者 reactive 来解决,而 toReftoRefs 则是把对象的数据进行分解和扩散,其这个对象针对的是响应式对象非普通对象总结起来有以下三点:

  • 初衷:不丢失响应式的情况下,把对象数据进行 分解或扩散
  • 前提: 针对的是响应式对象reactrive 封装的)而非普通对象
  • 注意: 不创造响应式,而是延续响应式。

二、🙆‍♀️Composition API实现逻辑复用

1、规则

先来了解几条规则:

  • Composition API抽离逻辑代码到一个函数
  • 函数的命名约定为 useXxxx 格式(React hooks也是);
  • setup 中引用 useXxx 函数。

2、举个例子🌰

引用一个非常经典的例子:获取鼠标的定位接下来我们用Composition API来进行封装演示:

定义一个 js 文件,名字为 useMousePosition具体代码如下:

import {
    reactive, ref, onMounted, onUnmounted } from 'vue'

function useMousePosition() {
   
    const x = ref(0)
    const y = ref(0)

    function update(e) {
   
        x.value = e.pageX
        y.value = e.pageY
    }

    onMounted(() => {
   
        console.log('useMousePosition mounted')
        window.addEventListener('mousemove', update)
    })

    onUnmounted(() => {
   
        console.log('useMousePosition unMounted')
        window.removeEventListener('mousemove', update)
    })

    return {
   
        x,
        y
    }
}

再定义一个 .vue 文件,命名为 index.vue具体代码如下:

<template>
    <p  v-if="flag">mouse position {
  {x}} {
  {y}}</p>
    <button @click="changeFlagHandler">change flag</button>
</template>

<script>
import {
     reactive } from 'vue'
import useMousePosition from './useMousePosition'

export default {
    
    name: 'MousePosition',
    return {
    
      flag: true
    },
    setup() {
    
        const {
     x, y } = useMousePosition()
        return {
    
            x,
            y
        }
    }changeFlagHandler() {
    
      this.flag = !this.flag
    },
}
</script>

此时浏览器的显示效果如下:

mousePosition

了解完 ref 后,我们来实现这个功能看起来会清晰很多。我们先通过 refxy 做响应式操作,之后通过 .value 来修改值,最终达到时刻获取鼠标定位的效果。同时,如果我们时刻保持着鼠标移动时不断改变值,这样子是非常耗费性能的。所以,我们可以通过一个按钮,来随时控制它的出现与隐藏。

大家可以发现,当隐藏的时候,随后会触发 onUnmounted 生命周期组件内容随之被销毁。也就是说,使用的时候调用,不使用的时候及时销毁,这样子可以很大程度上提升性能

三、🙅‍♀️结束语

通过上文的学习,我们可以知道, reftoReftoRefsvue3Composition API 的新特性,且 vue3 一般通过 reftoReftoRefs实现数据响应式。有了这三个内容,实现数据响应式看起来方便许多,而不再像 vue2 中那种处理起来很困难。

到这里,关于 reftoReftoRefs 的内容就讲完啦!希望对大家有帮助!

如有疑问或文章有误欢迎评论区浏览或私信我交流~

  • 关注公众号 星期一研究室 ,第一时间关注学习干货,更多有趣的专栏待你解锁~
  • 如果这篇文章对你有用,记得 一键三连 再走哦!
  • 我们下期见!🥂🥂🥂
相关文章
|
10天前
|
弹性计算 人工智能 架构师
阿里云携手Altair共拓云上工业仿真新机遇
2024年9月12日,「2024 Altair 技术大会杭州站」成功召开,阿里云弹性计算产品运营与生态负责人何川,与Altair中国技术总监赵阳在会上联合发布了最新的“云上CAE一体机”。
阿里云携手Altair共拓云上工业仿真新机遇
|
6天前
|
机器学习/深度学习 算法 大数据
【BetterBench博士】2024 “华为杯”第二十一届中国研究生数学建模竞赛 选题分析
2024“华为杯”数学建模竞赛,对ABCDEF每个题进行详细的分析,涵盖风电场功率优化、WLAN网络吞吐量、磁性元件损耗建模、地理环境问题、高速公路应急车道启用和X射线脉冲星建模等多领域问题,解析了问题类型、专业和技能的需要。
2506 14
【BetterBench博士】2024 “华为杯”第二十一届中国研究生数学建模竞赛 选题分析
|
6天前
|
机器学习/深度学习 算法 数据可视化
【BetterBench博士】2024年中国研究生数学建模竞赛 C题:数据驱动下磁性元件的磁芯损耗建模 问题分析、数学模型、python 代码
2024年中国研究生数学建模竞赛C题聚焦磁性元件磁芯损耗建模。题目背景介绍了电能变换技术的发展与应用,强调磁性元件在功率变换器中的重要性。磁芯损耗受多种因素影响,现有模型难以精确预测。题目要求通过数据分析建立高精度磁芯损耗模型。具体任务包括励磁波形分类、修正斯坦麦茨方程、分析影响因素、构建预测模型及优化设计条件。涉及数据预处理、特征提取、机器学习及优化算法等技术。适合电气、材料、计算机等多个专业学生参与。
1519 14
【BetterBench博士】2024年中国研究生数学建模竞赛 C题:数据驱动下磁性元件的磁芯损耗建模 问题分析、数学模型、python 代码
|
8天前
|
编解码 JSON 自然语言处理
通义千问重磅开源Qwen2.5,性能超越Llama
击败Meta,阿里Qwen2.5再登全球开源大模型王座
530 13
|
1月前
|
运维 Cloud Native Devops
一线实战:运维人少,我们从 0 到 1 实践 DevOps 和云原生
上海经证科技有限公司为有效推进软件项目管理和开发工作,选择了阿里云云效作为 DevOps 解决方案。通过云效,实现了从 0 开始,到现在近百个微服务、数百条流水线与应用交付的全面覆盖,有效支撑了敏捷开发流程。
19282 30
|
1月前
|
人工智能 自然语言处理 搜索推荐
阿里云Elasticsearch AI搜索实践
本文介绍了阿里云 Elasticsearch 在AI 搜索方面的技术实践与探索。
18836 20
|
1月前
|
Rust Apache 对象存储
Apache Paimon V0.9最新进展
Apache Paimon V0.9 版本即将发布,此版本带来了多项新特性并解决了关键挑战。Paimon自2022年从Flink社区诞生以来迅速成长,已成为Apache顶级项目,并广泛应用于阿里集团内外的多家企业。
17524 13
Apache Paimon V0.9最新进展
|
8天前
|
人工智能 自动驾驶 机器人
吴泳铭:AI最大的想象力不在手机屏幕,而是改变物理世界
过去22个月,AI发展速度超过任何历史时期,但我们依然还处于AGI变革的早期。生成式AI最大的想象力,绝不是在手机屏幕上做一两个新的超级app,而是接管数字世界,改变物理世界。
458 48
吴泳铭:AI最大的想象力不在手机屏幕,而是改变物理世界
|
1天前
|
云安全 存储 运维
叮咚!您有一份六大必做安全操作清单,请查收
云安全态势管理(CSPM)开启免费试用
354 4
叮咚!您有一份六大必做安全操作清单,请查收
|
2天前
|
存储 关系型数据库 分布式数据库
GraphRAG:基于PolarDB+通义千问+LangChain的知识图谱+大模型最佳实践
本文介绍了如何使用PolarDB、通义千问和LangChain搭建GraphRAG系统,结合知识图谱和向量检索提升问答质量。通过实例展示了单独使用向量检索和图检索的局限性,并通过图+向量联合搜索增强了问答准确性。PolarDB支持AGE图引擎和pgvector插件,实现图数据和向量数据的统一存储与检索,提升了RAG系统的性能和效果。