• # Vue3 toRef 和 toRefs 函数


    🚀 优质资源分享 🚀

    学习路线指引(点击解锁)知识定位人群定位
    🧡 Python实战微信订餐小程序 🧡进阶级本课程是python flask+微信小程序的完美结合,从项目搭建到腾讯云部署上线,打造一个全栈订餐系统。
    💛Python量化交易实战💛入门级手把手带你打造一个易扩展、更安全、效率更高的量化交易系统

    Vue3 toRef 和 toRefs 函数

    上一篇博文介绍了 vue3 里面的 ref 函数和 reactive 函数,实现响应式数据,今天主要来说一下 toRef 函数和 toRefs 函数的基本使用。

    toRef 函数

    通过上一篇博客,我们知道,ref 函数可以创建一个响应式的数据,那 toRef 函数同样也是创建一个响应式的数据,那么他们之间的区别是什么呢?

    首先一点,ref 函数他的本质其实是去拷贝一份数据,脱离了与源数据的交互。什么意思呢?就是 ref 函数可以将对象里面的属性值变成响应式的数据,修改响应式数据,是不会影响到源数据,但是视图层上的数据会被更新。但是 toRefs 函数的本质是引用,也就是说,toRef 函数会与源数据交互,修改响应式数据会造成源数据的修改,但是他的修改不会造成视图层数据的更新。

    上面这段话理解吗?不理解的话没关系,下面通过几个案例就可以明白了。

    toRef 函数使用

    首先呢, toRef 函数有两个参数。

    toRef(操作对象, 对象属性)
    
    
    • 1
    • 2

    好,接下来我们使用 toRef 函数写一个案例,还是和以前一样,页面展示一个用户的名称和年纪。

    
     <div>
     <h1>toRef toRefs 函数h1>
     <p>姓名:{{boy\_toRef}}p>
     <p>年龄:{{boy.age}}p>
     div>
    
    <script>
     import { toRef } from 'vue'
     export default {
     setup() {
     const boy = { // 创建一个用户对象
     name: '我是𝒆𝒅.', // 用户名称
     age: 10 // 用户年龄
     }
     // 使用 toRef 函数包裹,操作 boy 对象的 name 属性
     const boy\_toRef = toRef(boy, 'name') 
     console.log(boy\_toRef) // 我们直接打印看一下包裹后的数据格式
     return { boy, boy\_toRef }
     }
     }
    script>
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23

    保存代码,刷新页面。


    我们可以看到数据的结构,在 value 里面直接就是 boy 下面 name 的属性值,所以说,接下来我们编写一个按钮,点击按钮,修改这个 name 值。

    
     <div>
     <h1>toRef toRefs 函数h1>
     <p>姓名:{{boy\_toRef}}p>
     <p>年龄:{{boy.age}}p>
     <el-button type="primary" @click="btn">修改 nameel-button>
     div>
    
    <script>
     import { toRef } from 'vue'
     export default {
     setup() {
     const boy = {
     name: '我是𝒆𝒅.',
     age: 10
     }
     // 这个 boy\_toRef 就是被 toRef 函数操作过的 boy 的 name 值
     const boy\_toRef = toRef(boy, 'name') 
     const btn = () => {
     boy\_toRef.value = '𝒆𝒅.' // 把 name 修改成 𝒆𝒅.
     console.log(boy\_toRef) // 修改完成打印一下结果
     }
     return { boy, btn, boy\_toRef }
     }
     }
    script>
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27

    保存代码刷新页面,然后点击按钮看一下页面效果。

    通过截图展示的效果我们可以发现,boy_toRef 的值确实被修改了,但是呢,页面并没有改变,而且页面也没有出现错误。

    这是什么原因呢? 其实这不是 Bug 哈,在本篇博文开始就说过,toRef 函数会与源数据交互,修改响应式数据会造成源数据的修改,但是他的修改不会造成视图层数据的更新,所以说,这就是 toRef 函数的功能。确实,视图没有数据更新我们通过上面的截图看到了,但是源数据修改这个怎么看呢?没关系,在回答这个问题之前,我们首先得知道,什么是源数据。

    就像上面的代码:

    const boy = {
      name: '我是𝒆𝒅.',
      age: 10
    }
    const boy_toRef = toRef(boy, 'name')
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6

    toRef 函数将 boy 对象给包裹了起来,所以说,boy 对象就是源数据。

    所以说,想知道源数据有没有改变,在点击按钮之后,打印一下 boy 对象,看一下 boy 有没有被改变。

    
     <div>
     <h1>toRef toRefs 函数h1>
     <p>姓名:{{boy\_toRef}}p>
     <p>年龄:{{boy.age}}p>
     <el-button type="primary" @click="btn">修改 nameel-button>
     div>
    
    <script>
     import { toRef } from 'vue'
     export default {
     setup() {
     const boy = {
     name: '我是𝒆𝒅.',
     age: 10
     }
     const boy\_toRef = toRef(boy, 'name') // 这个 boy\_toRef 就是被 toRef 函数操作过的 boy 的 name 值
     const btn = () => {
     boy\_toRef.value = '𝒆𝒅.' // 把 name 修改成 𝒆𝒅.
     console.log(boy\_toRef) // 修改完成打印一下结果
     console.log(boy) // 修改完成打印一下boy结果
     }
     return { boy, btn, boy\_toRef }
     }
     }
    script>
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27

    保存代码,刷新页面,在点击按钮修改 name 值,然后查看一下控制台打印的 boy 对象。


    发现 boy 对象的 name 值已经从 我是𝒆𝒅. 改为 𝒆𝒅.了,但是页面依旧没有更新。

    记住了!

    toRef 函数会与源数据交互,修改响应式数据会造成源数据的修改,但是他的修改不会造成视图层数据的更新。

    ref 函数验证

    ref 函数可以将对象里面的属性值变成响应式的数据,修改响应式数据,是不会影响到源数据,但是视图层上的数据会被更新 这句话是正确的嘛?上一节我们没测试,所以说在这里我们也测试一下。

    我们还是写一个案例,页面展示一个名称,点击按钮,修改页面名称。

    
     <div>
     <h1>ref reactive 函数h1>
     <p>姓名:{{name\_ref}}p>
     <el-button type="primary" @click="btn">修改信息el-button>
     div>
    
    <script>
     import { ref } from 'vue'
     export default {
     setup() {
     const name = '我是𝒆𝒅.'
     const name\_ref = ref(name)
     const btn = () => {
     name\_ref.value = '𝒆𝒅.'
     console.log(name\_ref) // 打印一下被ref包裹的数据
     console.log(name) // 打印一下源数据
     }
     return { name\_ref, btn }
     }
     }
    script>
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23

    保存代码,刷新页面,点击按钮查看页面控制台打印的结果,主要是看一下被 ref 函数包裹后的数据有没有修改成功,源数据有没有修改成功,最后页面有没有修改,下面看截图。

    OK,通过上面截图,顾忌大家都理解了吧!

    所以再记住!

    ref 函数可以将对象里面的属性值变成响应式的数据,修改响应式数据,是不会影响到源数据,但是视图层上的数据会被更新

    toRefs 函数

    toRefs 函数的使用呢,其实和 toRef 函数类似的哈。

    • toRefs 函数用于批量设置多个数据为相应是数据。
    • toRefs 函数与原始数据相交互,修改响应式数据会影响到源数据,但是不会更新视图层。
    • toRefs 函数还可以与其他响应式数据相交互,更加方便处理视图层数据。

    toRefs 函数使用

    老样子,创建一个对象,然后使用 toRefs 函数包裹,在页面展示一下。

    
     <div>
     <h1>toRef toRefs 函数h1>
     <p>姓名:{{boy\_toRefs.name}}p>
     <p>年龄:{{boy\_toRefs.age}}p>
     div>
    
    <script>
     import { toRefs } from 'vue'
     export default {
     setup() {
     const boy = {
     name: '我是𝒆𝒅.',
     age: 10
     }
     const boy\_toRefs = toRefs(boy) // 将 boy 用 toRefs 包裹
     console.log(boy\_toRefs) // 打印一下结果
     return { boy\_toRefs }
     }
     }
    script>
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22

    保存代码,刷新页面查看。

    所以说,我们修改修改一下代码,在渲染的时候除了 .属性 之外,还需要 .value。

        姓名:{{boy\_toRefs.name.value}}
    
    
        <p>年龄:{{boy\_toRefs.age.value}}p>
    
    
    • 1
    • 2
    • 3
    • 4
    • 5

    把视图层代码修改一下,然后查看效果。

    诶,现在就是正常的啦!

    有人可能会疑问,那这玩意儿整的不是越来越复杂了吗?本来直接点属性就可以,现在还得点属性点value,不是多此一举,脱裤子放P吗? 嘿嘿嘿!我觉得也是。

    为什么呢说是多此一举也很正常,因为前面的博文讲过,这种复杂结构数据我们完全可以使用 reactive 函数来处理呀,渲染最多点一次就可以,但是 toRefs 函数却需要点两次。

    
     <div>
     <h1>toRef toRefs 函数h1>
     <p>姓名:{{boy\_toRefs.name}}p>
     <p>年龄:{{boy\_toRefs.age}}p>
     div>
    
    <script>
     import { toRefs, reactive } from 'vue'
     export default {
     setup() {
     const boy = {
     name: '我是𝒆𝒅.',
     age: 10
     }
     const boy\_toRefs = reactive(boy)
     return { boy\_toRefs }
     }
     }
    script>
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21

    我们不使用 toRefs 函数,而是用之前说的 reactive 函数处理数据。

    我们可以看到,页面是可以正常解析的,那为什么我们还有舍近求远的使用 toRefs 函数呢?

    其实是有原因的呀!

    其实 toRefs 函数最大的用处在这里!

    我们这个 boy 对象里面只有两个参数比较少,如果我们这个对象里面有十个参数或者是更多的话,每次展示的时候都得写那么多遍的 boy 点,是不是很麻烦呢?所以说使用 toRefs 函数就可以解决这个问题,看下面的代码。

    
     <div>
     <h1>toRef toRefs 函数h1>
     <p>姓名:{{name}}p>
     <p>年龄:{{age}}p>
     div>
    
    <script>
     import { toRefs } from 'vue'
     export default {
     setup() {
     const boy = {
     name: '我是𝒆𝒅.',
     age: 10
     }
     return { boy\_toRefs , ...toRefs(boy)}
     }
     }
    script>
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20

    在 return 抛出 reactive 的时候,使用扩展运算符和 toRefs 函数,就可以实现直接写属性的方式展示数据了。


    但是呢,深层次的对象依旧需要通过点来实现。

    也许你还有疑问,直接扩展运算 reactive 函数也行啊,为啥要套上 toRefs 函数,记住一点呀!

    toRefs 函数修改,原始数据被改变,页面不会被触发。

    看下面代码:

    
     <div>
     <h1>toRef toRefs 函数h1>
     <p>姓名:{{name}}p>
     <p>年龄:{{age}}p>
     <el-button type="primary" @click="btn">修改 nameel-button>
     div>
    
    <script>
     import { toRefs, reactive } from 'vue'
     export default {
     setup() {
     const boy = {
     name: '我是𝒆𝒅.',
     age: 10
     }
     const new\_toRefs = toRefs(boy)
    
     const btn = () => {
     new\_toRefs.name.value = '𝒆𝒅.'
     console.log(boy)
     }
    
     return { btn, ...toRefs(boy) }
     }
     }
    script>
    
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28

    打印一下结果:

    从打印结果中可以看出,原始数据被改变,页面没有被触发。但从我的写法上应该可以注意到,toRefs 返回的对象,随便解、随便构,丝毫不会影响值的响应性。

    总结

    有的小伙伴可能还是不太理解这两个函数,稍微总结一下子。

    • 如果想让响应式数据和以前的数据关联起来,并且想在更新响应式数据的时候不更新视图,那么就使用 toRef 函数。
    • 如果希望将对象的多个属性都变成响应式数据,并且要求响应式数据和原始数据关联,并且更新响应式数据的时候不更新视图,就使用 toRefs 函数用于批量设置多个数据为响应式数据。因为 toRef 函数一次仅能设置一个数据。
    • toRefs 函数接收一个对象作为参数,它会遍历对象身上的所有属性,然后挨个调用 toRef 函数执行。

    好了,今天的内容大体就是这些了,晚安宝子们,明天见!

    • 关于博主: 评论和私信会在第一时间回复。或者直接私信我。
    • 版权声明: 本博客所有文章除特别声明外,均采用 BY-NC-SA 许可协议。转载请注明出处!
    • 声援博主: 如果您觉得文章对您有帮助,可以点击文章右下角**【[推荐](javascript:void(0)😉】**一下。
  • 相关阅读:
    什么是 SSL?SSL/TLS是如何工作的?HTTP和HTTPS有什么区别?
    properties文件
    自定义可控自编码网络结构(一种新思路做AI绘画)
    时间序列论文-聚类和异常检测(一)
    postgresql参数优化
    Redis -- 分片集群
    windows 安装 Oracle Database 19c
    leetcode112.路径总和
    【ES专题】Logstash与FileBeat详解以及ELK整合详解
    MySQL之DQL
  • 原文地址:https://blog.csdn.net/xuhss_com/article/details/125612023