说一下 toRef、toRefs,以及他们的区别

2023-11-10 16:12
文章标签 区别 一下 toref torefs

本文主要是介绍说一下 toRef、toRefs,以及他们的区别,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

toRef:创建一个新的Ref变量,转换Reactive对象的某个字段为Ref变量
toRefs:创建一个新的对象,它的每个字段都是Reactive对象各个字段的Ref变量


说一下toRef
先定义一个reactive对象

interface Member {id: numbername: string
}
const userInfo: Member = reactive({id: 1,name: 'White'
})
console.log('userInfo=', userInfo)

如果想转换userInfo对象的name这个字段为Ref变量,可以进行如下操作:let nameR = toRef(userInfo, 'name')
此时,这个nameR是一个Ref变量,
所以,之后在读取和赋值时,就要使用nameR.value来操作,
所以,在重新给nameR赋值时,会同时更新nameRuserInfo.name的值

let nameR = toRef(userInfo, 'name')
console.log('nameR是一个Ref变量,值=', nameR.value)/**
* @name editNameR
* @description 修改 nameR 的值
*/
const editNameR = () => {nameR.value = 'edit-White'// 可以看到,在重新给`nameR`赋值后,`nameR` 和 `userInfo.name`的值都同时更新了console.log('edit - nameR=', nameR.value)console.log('edit - userInfo=', userInfo)
}

toRef也可以接收一个数组,此时第二个参数,是数组的下标

let wordList = reactive(['a', 'b', 'c'])
let a = toRef(wordList, 0)
console.log('a=', a.value) // a
console.log('wordList[0]=', wordList[0]) // a

对象 - 设置默认值
如果 Reactive 对象上有一个属性本身没有初始值,可以传递第三个参数进行设置(默认值仅对 Ref 变量有效)

interface MemberCopy {id: numbername: stringage?: number // age属性,因为是可选的,因此默认值会是`undefined`
}
// 声明变量时,省略`age`属性
const theInfo: MemberCopy = reactive({id: 1,name: 'Black'
})// 此时,为了避免程序运行错误,可以指定一个初始值,但初始值仅对 Ref 变量有效,不会影响 Reactive 字段的值
let age = toRef(theInfo, 'age', 18)
console.log('age=', age.value) // age= 18
console.log('theInfo.age=', theInfo.age) // theInfo.age= undefined// 除非重新赋值,才会使两者同时更新
age.value = 25
console.log(age.value) // 25
console.log(theInfo.age) // 25

数组 - 设置默认值

const words = reactive(['a', 'b', 'c'])// 当下标对应的值不存在时,返回`undefined`
const d = toRef(words, 3)
console.log(d.value) // undefined
console.log(words[3]) // undefined// 设置了默认值之后,就会对 Ref 变量使用默认值, Reactive 数组此时不影响
const e = toRef(words, 4, 'e')
console.log(e.value) // e
console.log(words[4]) // undefined

还有一个不推荐的特殊用法,
toRef 的过程中,如果使用了原对象上不存在的 key ,那么定义出来的 Ref变量的.value值将会是 undefined

举个例子
// 众所周知,White 是没有女朋友的
const girlfriend = toRef(userInfo, 'girl')
console.log('girlfriend=', girlfriend.value) // girlfriend= undefined
console.log('userInfo.girlfriend=', userInfo.girl) // userInfo.girl= undefined
// 此时 userInfo 对象上只有两个 Key
console.log(Object.keys(userInfo)) // ['id', 'name']/*如果,对这个不存在的`key的Ref变量(girlfriend)`进行赋值,那么原来的`Reactive对象(userInfo)`也会同步增加这个key(girl),其值也会同步更新。
*/
girlfriend.value = 'Marry'
console.log('girlfriend=', girlfriend.value) // girlfriend= Marry
console.log('userInfo.girl=', userInfo.girl) // userInfo.girlfriend= Marry
console.log('看看userInfo的属性=', userInfo) // Proxy的girl、id、name

为什么强调不要在 TypeScript 里使用呢?因为在编译时,无法通过 TypeScript 的类型检查
如果非用不可,可以考虑使用any类型,如下:1、2、3

// 1、将该类型直接指定为 `any`
type Member = any
// 当然一般都是 `const userInfo: any`// 2、或者保持接口类型的情况下,允许任意键值
interface Member {[key: string]: any
}// 3、使用 `Record` 也是同理
type Member = Record<string, any>



说一下toRefs:
与toRef不同,toRefs只接收一个参数(一个reactive变量)

interface People {id: numbername: string
}// 声明一个`Reactive`变量,此时 `theKing`的TS类型是:const theKing: People
const theKing: People = reactive({id: 1,name: 'Black'
})
console.log('theKing=', theKing)// 传给`toRefs`作为入参,此时,这个新的`useToRefs`变量的TS类型就不再是`People`了,而是:const useToRefs: ToRefs<People>
const useToRefs = toRefs(theKing)
console.log('useToRefs=', useToRefs)// 也可以重新编写一个新的类型来指定它,因为每个字段都是与原来关联的 Ref 变量,所以也可以这样声明:
interface newPeople {id: Ref<number>name: Ref<string>
}
const useToRefsCopy: newPeople = toRefs(theKing)
console.log('useToRefsCopy=', useToRefsCopy)话都到这了,其实,日常使用时并不需要手动指定其类型, TypeScript会自动推导,可以节约非常多的开发工作量

对数组进行转换

const charList = reactive(['a', 'b', 'c'])
const charListRefs = toRefs(charList)
console.log('charListRefs=', charListRefs)
console.log('charListRefs[0]=', charListRefs[0].value) // charListRefs[0]= a

解构与赋值,
这一点和直接解构 Reactive 变量有非常大的不同,直接解构 Reactive 变量,得到的是一个普通的变量,不再具备响应

// 用 `toRefs` 转换后的 `Reactive对象或数组` 支持ES6的解构,并且`不会失去响应性`,因为解构后的每一个变量都具备响应性。
const { name } = toRefs(theKing)
console.log('name=', name.value) // name= Black// 此时,对解构出来的变量重新赋值,原来的变量也会同步更新
name.value = 'Tom'
console.log('重新赋值后-name=', name.value) // 重新赋值后-name= Tom
console.log('重新赋值后-theKing', theKing.name) // 重新赋值后-theKing Tom// ---------------------------- 看下面这个例子 ----------------------------/*以一个计算器函数为例,这一次将其修改为内部有一个 Reactive 的数据状态中心,在函数返回时解构为多个 Ref 变量,这样在调用 useCalculator 函数时,可以通过解构直接获取到 Ref 变量,不需要再进行额外的转换工作。
*/
interface CalculatorState {num: number,step: number // 每次计算时要增加的幅度
}
// 声明一个 “使用计算器” 的函数
const useCalculator = () => {// 通过数据中心的形式,即中管理内部变量const state: CalculatorState = reactive({num: 0,step: 10})const add = () => {state.num += state.step}return {...toRefs(state),add}
}// 解构出来的 `num` 和 `step` 都是 Ref 变量
const { num, step, add } = useCalculator()
console.log('num=', num.value) // num= 0
console.log('step=', step.value) // step= 10
// 调用计算器的方法,数据也是会得到响应式更新
add()
console.log('调用add()方法之后,num=', num.value) // 调用add()方法之后,num= 10

这篇关于说一下 toRef、toRefs,以及他们的区别的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/383797

相关文章

关于Mybatis和JDBC的使用及区别

《关于Mybatis和JDBC的使用及区别》:本文主要介绍关于Mybatis和JDBC的使用及区别,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、JDBC1.1、流程1.2、优缺点2、MyBATis2.1、执行流程2.2、使用2.3、实现方式1、XML配置文件

exfat和ntfs哪个好? U盘格式化选择NTFS与exFAT的详细区别对比

《exfat和ntfs哪个好?U盘格式化选择NTFS与exFAT的详细区别对比》exFAT和NTFS是两种常见的文件系统,它们各自具有独特的优势和适用场景,以下是关于exFAT和NTFS的详细对比... 无论你是刚入手了内置 SSD 还是便携式移动硬盘或 U 盘,都需要先将它格式化成电脑或设备能够识别的「文

什么是ReFS 文件系统? ntfs和refs的优缺点区别介绍

《什么是ReFS文件系统?ntfs和refs的优缺点区别介绍》最近有用户在Win11Insider的安装界面中发现,可以使用ReFS来格式化硬盘,这是不是意味着,ReFS有望在未来成为W... 数十年以来,Windows 系统一直将 NTFS 作为「内置硬盘」的默认文件系统。不过近些年来,微软还在研发一款名

go 指针接收者和值接收者的区别小结

《go指针接收者和值接收者的区别小结》在Go语言中,值接收者和指针接收者是方法定义中的两种接收者类型,本文主要介绍了go指针接收者和值接收者的区别小结,文中通过示例代码介绍的非常详细,需要的朋友们下... 目录go 指针接收者和值接收者的区别易错点辨析go 指针接收者和值接收者的区别指针接收者和值接收者的

售价599元起! 华为路由器X1/Pro发布 配置与区别一览

《售价599元起!华为路由器X1/Pro发布配置与区别一览》华为路由器X1/Pro发布,有朋友留言问华为路由X1和X1Pro怎么选择,关于这个问题,本期图文将对这二款路由器做了期参数对比,大家看... 华为路由 X1 系列已经正式发布并开启预售,将在 4 月 25 日 10:08 正式开售,两款产品分别为华

kotlin中const 和val的区别及使用场景分析

《kotlin中const和val的区别及使用场景分析》在Kotlin中,const和val都是用来声明常量的,但它们的使用场景和功能有所不同,下面给大家介绍kotlin中const和val的区别,... 目录kotlin中const 和val的区别1. val:2. const:二 代码示例1 Java

CSS Padding 和 Margin 区别全解析

《CSSPadding和Margin区别全解析》CSS中的padding和margin是两个非常基础且重要的属性,它们用于控制元素周围的空白区域,本文将详细介绍padding和... 目录css Padding 和 Margin 全解析1. Padding: 内边距2. Margin: 外边距3. Padd

Springboot @Autowired和@Resource的区别解析

《Springboot@Autowired和@Resource的区别解析》@Resource是JDK提供的注解,只是Spring在实现上提供了这个注解的功能支持,本文给大家介绍Springboot@... 目录【一】定义【1】@Autowired【2】@Resource【二】区别【1】包含的属性不同【2】@

Java中的String.valueOf()和toString()方法区别小结

《Java中的String.valueOf()和toString()方法区别小结》字符串操作是开发者日常编程任务中不可或缺的一部分,转换为字符串是一种常见需求,其中最常见的就是String.value... 目录String.valueOf()方法方法定义方法实现使用示例使用场景toString()方法方法

分辨率三兄弟LPI、DPI 和 PPI有什么区别? 搞清分辨率的那些事儿

《分辨率三兄弟LPI、DPI和PPI有什么区别?搞清分辨率的那些事儿》分辨率这个东西,真的是让人又爱又恨,为了搞清楚它,我可是翻阅了不少资料,最后发现“小7的背包”的解释最让我茅塞顿开,于是,我... 在谈到分辨率时,我们经常会遇到三个相似的缩写:PPI、DPI 和 LPI。虽然它们看起来差不多,但实际应用