TypeScript-keyof 与 in 操作符

前言

keyofin操作符的使用方式看着有些相像,其实有很大的不同,keyof一般对对象类型使用,表示获取对象中的属性名。而in用于对象类型的定义中,右边跟一个联合类型,用来"迭代"联合类型简化对象类型的声明。本章介绍keyof和in操作符的用法,同时记录两个相关的实战案例。

一、keyof操作符的使用

1.1 基本使用

我们如果需要获取对象类型中的属性名,通常需要使用keyof。如下图,key可以使用"name"和"age"进行赋值,但是不能用"sex",因为Person类型中没有定义sex属性。

1.2 对对象索引签名使用

如果对声明了索引签名的对象使用keyof,会直接返回索引签名中的类型:

需要注意的是,如果索引签名的类型是string,那么使用keyof推导出的类型是number|string,原因是js中,使用数字作为属性名会被转换成字符串,SomeType[1] 和 SomeType["1"]是等价的。

1.3 对泛型对象使用

对泛型参数使用keyof时需要注意,在js中允许使用数字、字符串、和Symbol类型来作为属性名,因此对泛型类型和any类型使用keyof时,推导出的类型为string|number|symbol

如果确定了对象只有某个类型的属性名,可以使用Extract关键字将类型提取出来:

1.4 实战

商场做的会员等级分为C、B、A、S四个等级,用户购买之后根据出示的会员卡计算折扣

根据描述我们很容易能写出如下的式子,但是ts会提示discountStrategy有确定的属性名,传入string是不合法的。

这个时候就可以借助keyof来获取discountStrategy的属性名,并声明VIPLevel只属于这个属性名范围:

实现步骤如下:

  1. 我们首先用typeof discountStrategy来获取discountStrategy的类型
  2. 再用keyof获取该类型的所有属性名
  3. 使用extends约束泛型T的对象上需要有discountStrategy的类型对于的属性名
  4. 最后将泛型T赋值给VIPLevel参数

二、in操作符的使用

in操作符比较简单,一般用在对象类型的声明中, 右边跟一个联合类型。常用方法如下:

可以看到,使用了in关键字后,ts自动推断出了CarValue的属性名和类型,不需要我们手动写。

上一章的实战使用in操作符,我们可以进行如下的优化:

实现步骤如下:

  1. 我们首先定义一个VIPLevel联合类型,列举所有会员等级
  2. 再申明一个VIPDiscountStrategy对象类型,使用in操作符来统一定义这个对象的属性值
  3. 最后在calculate函数上将参数VIPLevel的类型声明成VIPLevel联合类型即可实现相同效果

三、综合实战

原题地址: 实现Pick关键字

具体实现:

scala 复制代码
type MyPick<T, K extends keyof T> = {
  [key in K]: T[key];
};

实现思路:

  1. 我们定义两个泛型参数T和K,其中约束K的类型扩展T的属性名
  2. 再用key in K: T[key]来取出Pick出来的属性名

使用示例:

相关推荐
转转技术团队2 分钟前
HLS 流媒体技术:畅享高清视频,忘却 MP4 卡顿的烦恼!
前端
程序员的程6 分钟前
我做了一个前端股票行情 SDK:stock-sdk(浏览器和 Node 都能跑)
前端·npm·github
KlayPeter7 分钟前
前端数据存储全解析:localStorage、sessionStorage 与 Cookie
开发语言·前端·javascript·vue.js·缓存·前端框架
沉默-_-9 分钟前
从小程序前端到Spring后端:新手上路必须理清的核心概念图
java·前端·后端·spring·微信小程序
裴嘉靖12 分钟前
前端获取二进制文件并预览的完整指南
前端·pdf
李剑一12 分钟前
uni-app使用瓦片实现离线地图的两种方案
前端·trae
木易 士心13 分钟前
深入剖析:按下 F5 后,浏览器前端究竟发生了什么?
前端
几何心凉15 分钟前
离开舒适区之后:从三年前端到 CS 硕士——我在韩国亚大读研的得失
前端·人工智能·年度总结
小二·19 分钟前
前端测试体系完全指南:从 Vitest 单元测试到 Cypress E2E(Vue 3 + TypeScript)
前端·typescript·单元测试
pas13620 分钟前
18-mini-vue element
前端·vue.js·ubuntu