ref、toRef、和toRefs傻傻分不清楚

Vue3响应式数据:ref、toRef与toRefs的深度解析
本文介绍了Vue3中ref、toRef和toRefs的使用,强调了它们的区别。ref创建的响应式数据是拷贝,不直接影响原始数据,而toRef创建的响应式数据与原始数据有引用关系,修改会同步。toRefs则用于批量创建对象属性的响应式数据。在使用时,ref会自动更新UI,而toRef不会。了解这些区别有助于更好地管理和更新Vue3应用的状态。

前言

这篇我们看下ref、toRef和toRefs的基本使用

我们知道ref可以用于创建一个响应式数据,而toRef也可以创建一个响应式数据,那他们之间有什么区别呢?
事实上,如果利用ref函数将某个对象中的属性变成响应式数据,修改响应式数据是不会影响到原始数据。


  • ref

import { ref } from 'vue';
export default {
  name:'App'
  setup(){
    let obj = {name : 'alice', age : 12};
    let newObj= ref(obj.name);
    const change = () => {
      newObj.value = 'Tom';
      console.log(obj,newObj)
    }
    return {newObj,change}
  }
}

上述代码,当change执行的时候,响应式数据发生改变,而原始数据obj并不会改变。

原因在于,ref的本质是拷贝,与原始数据没有引用关系
  • toRef

而如果使用toRef将某个对象中的属性变成响应式数据,修改响应式数据是会影响到原始数据的。但是需要注意,如果修改通过toRef创建的响应式数据,并不会触发UI界面的更新。

所以,toRef的本质是引用,与原始数据有关联
import {toRef} from 'vue';
export default {
  name:'App'
  setup(){
    let obj = {name : 'alice', age : 12};
    let newObj= toRef(obj, 'name');
    const change = () => {
      newObj.value = 'Tom';
      console.log(obj,newObj)
    }
    return {newObj,change}
  }
}

上述代码,当change执行的时候,响应式数据发生改变,原始数据obj并不会改变,但是UI界面不会更新

小结:
ref和toRef的区别
(1). ref本质是拷贝,修改响应式数据不会影响原始数据;toRef的本质是引用关系,修改响应式数据会影响原始数据
(2). ref数据发生改变,界面会自动更新;toRef当数据发生改变是,界面不会自动更新
(3). toRef传参与ref不同;toRef接收两个参数,第一个参数是哪个对象,第二个参数是对象的哪个属性

所以如果想让响应式数据和以前的数据关联起来,并且想在更新响应式数据的时候不更新UI,那么就使用toRef

  • toRefs

有的时候,我们希望将对象的多个属性都变成响应式数据,并且要求响应式数据和原始数据关联,并且更新响应式数据的时候不更新界面,就可以使用toRefs,用于批量设置多个数据为响应式数据。(toRef一次仅能设置一个数据)
toRefs接收一个对象作为参数,它会遍历对象身上的所有属性,然后挨个调用toRef执行

import {toRefs} from 'vue';
export default {
  name:'App'
  setup(){
    let obj = {name : 'alice', age : 12};
    let newObj= toRefs(obj);
    const change = () => {
      newObj.name.value = 'Tom';
      newObj.age.value = 18;
      console.log(obj,newObj)
    }
    return {newObj,change}
  }
}
  • toRefs解决return变量过多的问题

在setup中定义的变量如果想要在HTML模板中使用,就需要将数据return出去。如果数据过多就需要在return中再写一遍这个数据。

import {toRefs} from 'vue';
export default {
  name:'App'
  setup(){
    let obj = reactive({
		dataList: [],
		isLoading: false,
		handleChange: ()=>{
			....
		}
	});
    return { ...toRefs(obj) }
  }
}

总结

以上就是vue3 ref、toRef函数和toRefs函数的基本使用。

`ref`、`toRef` `toRefs` 都是 Vue 3 中用于处理响应式数据的工具,它们之间有以下区别: ### 功能用途 - **ref**:用于创建一个值类型的响应式数据,可用于模板,并且可以通过 `.value` 修改值。它既可以用来包装基本数据类型(如数字、字符串等),也能包装对象或数组,将其转换为响应式数据。例如: ```vue <template> <div>{{ count.value }}</div> <button @click="increment">Increment</button> </template> <script setup> import { ref } from 'vue'; const count = ref(0); const increment = () => { count.value++; }; </script> ``` - **toRef**:针对一个响应式对象(`reactive` 创建的)的某个属性创建一个 `ref`,两者保持引用关系。当需要单独处理某个响应式属性时使用,例如要将响应式对象中的某个属性单独提供给外部使用时。示例代码如下: ```vue <template> <div>{{ name.value }}</div> <button @click="changeName">Change Name</button> </template> <script setup> import { reactive, toRef } from 'vue'; const person = reactive({ name: 'John' }); const name = toRef(person, 'name'); const changeName = () => { name.value = 'Jane'; }; </script> ``` - **toRefs**:将一个响应式对象(`reactive` 封装的)转换为普通对象,对象的每个属性都是对应的 `ref`,同样与原始响应式对象保持引用关系。适用于需要将整个响应式对象的属性都转换为 `ref` 的场景,通常用于将响应式对象的属性传递给其他组件或函数时。示例代码如下: ```vue <template> <div>{{ personRefs.name.value }}</div> <button @click="changeName">Change Name</button> </template> <script setup> import { reactive, toRefs } from 'vue'; const person = reactive({ name: 'John' }); const personRefs = toRefs(person); const changeName = () => { personRefs.name.value = 'Jane'; }; </script> ``` ### 创建对象的目标 - **ref**:可以基于基本数据类型或对象直接创建响应式引用。 - **toRef toRefs**:必须针对一个响应式对象(`reactive` 创建的),普通对象无法通过它们变成响应式的,其设计初衷就是针对响应式对象进行操作 [^4]。 ### 创建数量 - **ref**:一次只能创建一个响应式引用。 - **toRef**:一次只能为响应式对象的一个属性创建 `ref`。 - **toRefs**:可以批量为响应式对象的所有属性创建 `ref` [^3]。 ### 同步性及注意事项 - **ref**:是一个独立的响应式引用,其值的修改不依赖于其他对象。 - **toRef toRefs**:创建的 `ref` 都与原始响应式对象保持同步,修改 `ref` 的值会影响原始对象,反之亦然。不过在使用 `toRefs` 时,如果原始响应式对象的属性发生变化(如添加新属性),`toRefs` 返回的对象不会自动更新,需要重新调用 `toRefs` 来获取最新的 `ref` 属性 [^1]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值