说一下 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赋值时,会同时更新nameR 和 userInfo.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,以及他们的区别
toRef:创建一个新的Ref变量,转换Reactive对象的某个字段为Ref变量 toRefs:创建一个新的对象,它的每个字段都是Reactive对象各个字段的Ref变量 说一下toRef 先定义一个reactive对象 interface Member {id: numbername: string } c…...
修改Android Studio默认的gradle目录
今天看了一下,gradle在C盘占用了40多G。我C盘是做GHOST的,放在这里不方便。所以就要修改。 新建目录名(似乎无必要) ANDROID_SDK_HOMEG:\SOFTWARES\android-sdk GRADLE_USER_HOMEG:\SOFTWARES\.gradle 修改目录 File->Setti…...
鲁大师电动车智能化测评报告第二十三期:实测续航95km,九号Q90兼顾个性与实用
鲁大师第二十三期智能化电动车测评排行榜数据来源于鲁大师智慧实验室,测评的车型均为市面上主流品牌的主流车型。截止目前,鲁大师智能化电动车测评的车型高达130余台,且还在不断增加和丰富中。 一、测评依据 鲁大师电动车智能化测评体系包含车辆的状态采集与管理硬件系统、车辆…...
初始化项目骨架(Web3项目一实战之一)
暌违将近一年的时光,也该是时候来几个项目实践。要不,当再次翻看 玩以太坊链上项目的必备技能(…solidity之旅X) ,却未曾见有关于 Web3 项目的实战博文,不免让人唏嘘! 其实,在我敲下玩以太坊链上项目的必备技能这些文字时,心中早有了势必要弄出一个Web3的项目(当然,通…...
在opencv OpenCV中打开相机摄像头,用分水岭算法实时实现图像的分割与提取
import cv2 import numpy as np# 定义回调函数 def callback(x):pass# 打开摄像头 cap cv2.VideoCapture(0)# 创建窗口和控件 cv2.namedWindow(image) cv2.createTrackbar(threshold, image, 0, 255, callback)# 初始化参数 bgdModel np.zeros((1, 65), np.float64) fgdModel…...
CodeWhisperer 的正确使用
1、重点: 重点1: 推出 Amazon Bedrock。这项新服务允许用户通过 API 访问来自 AI21 Labs、Anthropic、Stability AI 和亚马逊的基础模型。(Anthropic 就是之前跟 ChatGPT 掰手腕的 Claude 的模型。Stability AI 就是 Stable Diffusion 背后的…...
selenium xpath定位
selenium-xpath定位 <span style"background-color:#2d2d2d"><span style"color:#cccccc"><code class"language-javascript">element_xpath <span style"color:#67cdcc"></span> driver<span styl…...
「我在淘天做技术」音视频技术及其在淘宝内容业务中的应用
作者:李凯 一、前言 近年来,内容电商似乎已经充分融入到人们的生活中:在闲暇时间,我们已经习惯于拿出手机,从电商平台的直播间、或者短视频链接下单自己心仪的商品。 尽管优质的货品、实惠的价格、精致的布景、有趣的…...
el-input 输入后失去焦点
说了是无情,写了更无情,你说你看了不点赞是不是更绝情?遇到这种神奇的BUG,也是大家无奈的神情。 来分析看代码: <div class"card-item input-item" :class"{ w-100: followRadio 2 }"v-for&…...
docker创建并访问本地前端
docker创建并访问本地前端,直接上命令: 安装nginx镜像: docker pull nginx 查看已安装的nginx: docker images 创建DockerFile文件,直接在当前文件夹种创建 touch Dockerfile 在Dockerfile写入内容: F…...
数据结构之单链表基本操作
🤷♀️🤷♀️🤷♀️ 今天给大家分享的是单链表的基本操作。 清风的个人主页 🎉欢迎👍点赞✍评论❤️收藏 😛😛😛希望我的文章能对你有所帮助,有不足的地方还请各位…...
Python 实践
文章目录 一、HttpRequests 一、Http Requests python——Request模块...
使用easyui前端框架快速构建一个crud应用
本篇文章将会详细介绍jquery easyui前端框架的使用,通过创建一个crud应用来带大家快速掌握easyui的使用。 easyui是博主最喜欢的前端框架,没有之一,因为它提供了多种主题,而且有圆润的各种组件。 一、快速开始 easyui的官网地址&…...
Logback从添加依赖,到配置给中打印级别,archive相关信息配置,在项目中的常见的用法,一个完整的过程
添加Logback依赖: 在您的Maven或Gradle项目中,添加Logback依赖。例如,在Maven中,可以将以下依赖添加到pom.xml文件中: <dependency><groupId>ch.qos.logback</groupId><artifactId>logback-c…...
虚假内容检测,谣言检测,不实信息检测,事实核查;纯文本,多模态,多语言;数据集整理
本博客系博主个人理解和整理所得,包含内容无法详尽,如有补充,欢迎讨论。 这里只提供数据集相关介绍和来源出处,或者下载地址等,因版权原因不提供数据集所含的元数据。如有需要,请自行下载。 “Complete d…...
数据结构:单链表
文章目录 🍉前言🍉基本概念🍉链表的分类🍌单链表节点的结构🍌创建节点🍌打印链表🍌插入和删除🥝尾插🥝头插🥝尾删🥝头删🥝指定位置之前…...
官媒代运营:让大众倾听品牌的声音
在当今数字时代,媒体的影响力和多样性远远超出了以往的范畴。品牌和企业越来越依赖媒体来传播信息、建立声誉以及与大众互动。而媒体矩阵成为了现代品牌传播的关键策略,使大众能够倾听品牌的声音。媒体矩阵:多元化的传播渠道 媒体矩阵是指利…...
postgresql 实现计算日期间隔排除周末节假日方案
前置条件:需要维护一张节假日日期表。例如创建holiday表保存当年假期日期 CREATE TABLE holiday (id BIGINT(10) ZEROFILL NOT NULL DEFAULT 0,day TIMESTAMP NULL DEFAULT NULL,PRIMARY KEY (id) ) COMMENT假期表 COLLATEutf8mb4_0900_ai_ci ;返回日期为xx日xx时x…...
金融工作怎么做?低代码如何助力金融行业
10月30日至31日,中央金融工作会议在北京举行。金融是国民经济的“血脉”,是国家核心竞争力的重要组成部分。会议指出,党的十八大以来,在党中央集中统一领导下,金融系统有力支撑经济社会发展大局,坚决打好防…...
基于springboot实现智慧外贸平台系统【项目源码+论文说明】计算机毕业设计
基于springboot实现智慧外贸平台系统演示 摘要 网络的广泛应用给生活带来了十分的便利。所以把智慧外贸管理与现在网络相结合,利用java技术建设智慧外贸平台,实现智慧外贸的信息化。则对于进一步提高智慧外贸管理发展,丰富智慧外贸管理经验能…...
别再只盯着铜箔了!FPC软板选材实战:从PI基材到屏蔽膜,工程师避坑指南
FPC软板选材实战:从基材到屏蔽层的工程决策指南 在可穿戴设备折叠屏和车载摄像头小型化的浪潮中,柔性印刷电路板(FPC)正经历前所未有的技术迭代。当某头部TWS耳机厂商因基材选择失误导致批量性断裂时,当新能源汽车摄像头模组因屏蔽材料失效引…...
【Flink实战指南】基于Table API与SQL Client的Catalog统一管理实践
1. 为什么需要统一管理Catalog? 在Flink的实际应用中,我们经常会遇到这样的场景:数据分散在不同的存储系统中,比如Hive、MySQL、Kafka等。每次操作这些数据时,都需要手动指定对应的连接信息,不仅效率低下&a…...
如何快速安装QuantEcon.py:完整环境配置教程
如何快速安装QuantEcon.py:完整环境配置教程 【免费下载链接】QuantEcon.py A community based Python library for quantitative economics 项目地址: https://gitcode.com/gh_mirrors/qu/QuantEcon.py QuantEcon.py是一个基于社区的Python定量经济学库&…...
【Docker 27网络策略终极指南】:27项生产级策略配置、隔离与审计实战(附策略合规检查清单)
第一章:Docker 27网络策略演进与核心架构解析Docker 27(代号“Nexus”)标志着容器网络模型的一次范式跃迁,其网络策略体系不再仅围绕桥接、主机与覆盖网络的静态划分,而是以零信任原则为基底,将策略执行点下…...
告别‘看图说话’式假新闻:用HAMMER模型实战检测图文双模态篡改(附数据集与代码)
实战HAMMER模型:从零构建图文双模态篡改检测系统 在信息爆炸的时代,图文并茂的"新闻"往往比纯文字更具传播力,也更容易让人信以为真。但你是否想过,那些看似真实的明星声明截图或政治人物"发言",可…...
【电力系统】基于萤火虫算法FA的太阳能风能水力混合抽水蓄能系统(Matlab代码实现)
💥💥💞💞欢迎来到本博客❤️❤️💥💥 🏆博主优势:🌞🌞🌞博客内容尽量做到思维缜密,逻辑清晰,为了方便读者。 ⛳️座右铭&a…...
逆向分析第一课:拆解Cheat Engine Tutorial,理解程序内存与汇编指令的互动
逆向工程入门:通过Cheat Engine Tutorial透视程序内存与汇编的奥秘 当第一次打开Cheat Engine时,许多用户会被其看似复杂的界面所震慑——十六进制数值、内存地址、汇编指令这些术语仿佛在构建一道技术壁垒。但正是这套工具,为我们打开了一扇…...
从手机拍照到NeRF建模:相机标定参数(内参/外参)到底在忙活啥?
从手机拍照到NeRF建模:相机标定参数(内参/外参)到底在忙活啥? 当你用手机拍下一张照片时,是否注意到画面边缘的直线有时会弯曲?或者在使用AR应用时,虚拟物体为何能稳稳"坐"在桌面上&a…...
告别Postman!用Apifox测试套件搞定团队接口自动化(附CI/CD集成实战)
从Postman迁移到Apifox:打造高效团队接口自动化测试体系 在DevOps和持续交付成为主流的今天,接口自动化测试已成为研发流程中不可或缺的一环。传统方案如PostmanNewman虽然广为人知,但在团队协作、版本管理和CI/CD集成方面存在明显短板。Apif…...
ThinkPHP 通用的API格式封装实例代码
ThinkPHP 通用的API格式封装1.创建status.php 用于设置通用的状态码返回枚举类1234567<?phpreturn["success">1,"error">0,"controller_not_found">-1,"action_not_found">-2,];2.将API返回格式统一封装1234567891011…...
