工具类型是 Typescript 附带的特殊类型,可用于提高代码的可读性和灵活性。简单地说,根据提供的类型,工具类型将会按照规则构造一个新类型。下面就来看看TypeScript中有哪些常用的工具类型以及使用方式!
Partial
作用是将传入的属性变为可选项。适用于对类型结构不明确的情况。它使用了两个关键字:keyof
和in
,先来看看它们都是什么含义。keyof
可以用来取得接口的所有 key
值:
type Person = {
name: string;
age: number;
height: number;
}
type T = keyof Person
// T 类型为: "name" | "age" | "number"
in
关键字可以遍历枚举类型,:
type Person = "name" | "age" | "number"
type Obj = {
[p in Person]: any
}
// Obj 的类型为: { name: any, age: any, number: any }
keyof
可以产生联合类型, in
可以遍历枚举类型。所以可以一起使用, 下面是Partial
的定义:
/**
* Make all properties in T optional
* 将T中的所有属性设置为可选
*/
type Partial<T> = {
[P in keyof T]?: T[P];
};
这里,keyof T
用来获取 T
所有属性名, 然后使用 in
进行遍历, 将值赋给 P
, 最后 T[P]
取得相应属性的值。中间的?
就用来将属性设置为可选。
来看下面的例子:
type Person = {
name: string;
age: number;
height: number;
}
type PartialPerson = Partial<Person>;
// PartialPerson 的类型为 {name?: string; age?: number; height?: number;}
const person: PartialPerson = {
name: "zhangsan";
}
这里就使用Partial
将Person
类型中的属性都指定为可选属性。
Required
的作用是将传入的属性变为必选项,和上面的Partial
恰好相反,其声明如下:
/**
* Make all properties in T required
* 将T中的所有属性设置为必选
*/
type Required<T> = {
[P in keyof T]-?: T[P];
};
可以看到,这里使用-?
将属性设置为必选,可以理解为减去问号。使用形式和上面的Partial
差不多:
type Person = {
name?: string;
age?: number;
height?: number;
}
type RequiredPerson = Required<Person>;
// RequiredPerson 的类型为 {name: string; age: number; height: number;}
const person: RequiredPerson = {
name: "zhangsan";
age: 18;
height: 180;
}
这里就使用Required
将Person
类型中的属性都指定为必选属性。
将T类型的所有属性设置为只读(readonly
),构造出来类型的属性不能被再次赋值。Readonly
的声明形式如下:
/**
* Make all properties in T readonly
*/
type Readonly<T> = {
readonly [P in keyof T]: T[P];
};
来看下面的例子:
type Person = {
name: string;
age: number;
}
type ReadonlyPerson = Readonly<Person>;
const person: ReadonlyPerson = {
name: "zhangsan",
age: 18
}
person.age = 20; // Error: cannot reassign a readonly property
可以看到,通过 Readonly
将Person
的属性转化成了只读,不能再进行赋值操作。Readonly
类型对于冻结对象非常有用。
从 Type
类型中挑选部分属性 Keys
来构造新的类型。它的声明形式如下:
/**
* From T, pick a set of properties whose keys are in the union K
*/
type Pick<T, K extends keyof T> = {
[P in K]: T[P];
};
来看下面的例子:
type Person = {
name: string;
age: number;
height: number;
}
const person: Pick<Person, "name" | "age"> = {
name: "zhangsan",
age: 18
}
这样就使用Pick
从Person
类型中挑出来了name
和age
属性的类型,新的类型中只包含这两个属性。
Record
用来构造一个类型,其属性名的类型为Keys
中的类型,属性值的类型为Type
。这个工具类型可用来将某个类型的属性映射到另一个类型上,下面是其声明形式:
/**
* Construct a type with a set of properties K of type T
*/
type Record<K extends keyof any, T> = {
[P in K]: T;
};
来看下面的例子:
type Pageinfo = {
title: string;
}
type Page = 'home' | 'about' | 'contact';
const page: Record<Page, Pageinfo> = {
about: {title: 'about'},
contact: {title: 'contact'},
home: {title: 'home'},
}
Exclude
用于从类型Type
中去除不在ExcludedUnion
类型中的成员,下面是其声明的形式:
/**
* Exclude from T those types that are assignable to U
*/
type Exclude<T, U> = T extends U ? never : T;
来看下面的例子:
type Person = {
name: string;
age: number;
height: number;
}
const person: Exclude<Person, "age" | "sex"> = {
name: "zhangsan";
height: 180;
}
这里就使用Exclude
将Person
类型中的age
属性给剔除了,只会剔除两个参数中都包含的属性。
Extract
用于从类型Type
中取出可分配给Union
类型的成员。作用与Exclude相反。下面是它的声明形式:
/**
* Extract from T those types that are assignable to U
*/
type Extract<T, U> = T extends U ? T : never;
来看下面的例子:
type ExtractedType = Extract<"x" | "y" | "z", "x" | "y">;
// "x" | "y"
该工具类型对于找出两种类型的公共部分很有用:
interface Human {
id: string;
name: string;
surname: string;
}
interface Cat {
id: string;
name: string;
sound: string;
}
// "id" | "name"
type CommonKeys = Extract<keyof Human, keyof Cat>;
上面的 Pick
和 Exclude
都是最基础的工具类型,很多时候用 Pick
或者 Exclude
可能不如直接写类型更直接。而 Omit 就基于这两个来做的一个更抽象的封装,它允许从一个对象中剔除若干个属性,剩下的就是需要的新类型。下面是它的声明形式:
/**
* Construct a type with the properties of T except for those in type K.
*/
type Omit<T, K extends keyof any> = Pick<T, Exclude<keyof T, K>>;
来看下面的例子:
type Person = {
name: string;
age: number;
height: number;
}
const person: Omit<Person, "age" | "height"> = {
name: "zhangsan";
}
这样就使用Omit
从Person
类型中剔除了 age
和 height
属性,只剩下 name
属性。
ReturnType
会返回函数返回值的类型,其声明形式如下:
/**
* Obtain the return type of a function type
*/
type ReturnType<T extends (...args: any) => any> = T extends (...args: any) => infer R ? R : any;
来看下面的例子:
function foo(type): boolean {
return type === 0
}
type FooType = ReturnType<typeof foo>
这里使用 typeof
是为了获取 foo
的函数签名,等价于 (type: any) => boolean
。
InstanceType 会返回 Type 构造函数类型的实例类型。其声明形式如下:
/**
* Obtain the return type of a constructor function type
*/
type InstanceType<T extends abstract new (...args: any) => any> = T extends abstract new (...args: any) => infer R ? R : any;
来看下面的例子:
class Person {
name: string;
age: number;
constructor(person: { name: string; age: number }) {
this.name = person.name;
this.age = person.age;
}
}
type PersonInstanceType = InstanceType<typeof Person>;
// PersonInstanceType 的类型:{ name: string; age: number }
当然,你可能不会这么写,因为可以直接使用UserManager
类型:
class Person {
name: string;
age: number;
constructor(person: { name: string; age: number }) {
this.name = person.name;
this.age = person.age;
}
}
const person: Person = {
name: "zhangsan",
age: 18,
};
这就等价于:
class Person {
name: string;
age: number;
constructor(person: { name: string; age: number }) {
this.name = person.name;
this.age = person.age;
}
}
type PersonInstanceType = InstanceType<typeof Person>;
const person: PersonInstanceType = {
name: "zhangsan",
age: 18,
};
当我们在 TypeScript 中创建动态类时,InstanceType
可以用于检索动态实例的类型。
Parameters
可以从函数类型Type
的参数中使用的类型构造一个元组类型。其声明形式如下:
/**
* Obtain the parameters of a function type in a tuple
*/
type Parameters<T extends (...args: any) => any> = T extends (...args: infer P) => any ? P : never;
来看下面的例子:
const add = (x: number, y: number) => {
return x + y;
};
type FunctionParameters = Parameters<typeof add>;
// FunctionParameters 的类型:[x: number, y: number]
除此之外,还可以检测单个参数:
// "number"
type FirstParam = Parameters<typeof add>[0];
// "number"
type SecondParam = Parameters<typeof add>[1];
// "undefined"
type ThirdParam = Parameters<typeof add>[2];
Parameters 对于获取函数参数的类型以确保类型安全很有用,尤其是在使用第三方库时:
const saveUser = (user: { name: string; height: number; age: number }) => {
// ...
};
const user: Parameters<typeof saveUser>[0] = {
name: "zhangsan",
height: 180,
age: 18,
};
ConstructorParameters
可以从构造函数的类型来构造元组或数组类型。其声明形式如下:
/**
* Obtain the parameters of a constructor function type in a tuple
*/
type ConstructorParameters<T extends abstract new (...args: any) => any> = T extends abstract new (...args: infer P) => any ? P : never;
它类似于参数,但适用于类构造函数:
class Person {
private name: string;
private age: number;
constructor(person: { name: string; age: number }) {
this.name = person.name;
this.age = person.age;
}
}
type ConstructorParametersType = ConstructorParameters<typeof Person>;
// ConstructorParametersType 的类型:[person: { name: string, age: number}]
与 Parameters
类型一样,当使用外部库时,它有助于确保构造函数接受我们传入的参数:
class Person {
private name: string;
private age: number;
constructor(person: { name: string; age: number }) {
this.name = person.name;
this.age = person.age;
}
}
const params: ConstructorParameters<typeof Person>[0] = {
name: "zhangsan",
age: 18,
};
NonNullable
通过从Type
中排除null
和undefined
来创建新类型。它就等价于Exclude<T, null | undefined>
。其声明形式如下:
/**
* Exclude null and undefined from T
*/
type NonNullable<T> = T extends null | undefined ? never : T;
来看下面的例子:
type Type = string | null | undefined;
// string
type NonNullableType = NonNullable<Type>;
这里就使用NonNullable
将Type
中的null
和undefined
剔除掉了。
本文由哈喽比特于2年以前收录,如有侵权请联系我们。
文章来源:https://mp.weixin.qq.com/s/lL0yISTn4bvqSixrp92nLw
京东创始人刘强东和其妻子章泽天最近成为了互联网舆论关注的焦点。有关他们“移民美国”和在美国购买豪宅的传言在互联网上广泛传播。然而,京东官方通过微博发言人发布的消息澄清了这些传言,称这些言论纯属虚假信息和蓄意捏造。
日前,据博主“@超能数码君老周”爆料,国内三大运营商中国移动、中国电信和中国联通预计将集体采购百万台规模的华为Mate60系列手机。
据报道,荷兰半导体设备公司ASML正看到美国对华遏制政策的负面影响。阿斯麦(ASML)CEO彼得·温宁克在一档电视节目中分享了他对中国大陆问题以及该公司面临的出口管制和保护主义的看法。彼得曾在多个场合表达了他对出口管制以及中荷经济关系的担忧。
今年早些时候,抖音悄然上线了一款名为“青桃”的 App,Slogan 为“看见你的热爱”,根据应用介绍可知,“青桃”是一个属于年轻人的兴趣知识视频平台,由抖音官方出品的中长视频关联版本,整体风格有些类似B站。
日前,威马汽车首席数据官梅松林转发了一份“世界各国地区拥车率排行榜”,同时,他发文表示:中国汽车普及率低于非洲国家尼日利亚,每百户家庭仅17户有车。意大利世界排名第一,每十户中九户有车。
近日,一项新的研究发现,维生素 C 和 E 等抗氧化剂会激活一种机制,刺激癌症肿瘤中新血管的生长,帮助它们生长和扩散。
据媒体援引消息人士报道,苹果公司正在测试使用3D打印技术来生产其智能手表的钢质底盘。消息传出后,3D系统一度大涨超10%,不过截至周三收盘,该股涨幅回落至2%以内。
9月2日,坐拥千万粉丝的网红主播“秀才”账号被封禁,在社交媒体平台上引发热议。平台相关负责人表示,“秀才”账号违反平台相关规定,已封禁。据知情人士透露,秀才近期被举报存在违法行为,这可能是他被封禁的部分原因。据悉,“秀才”年龄39岁,是安徽省亳州市蒙城县人,抖音网红,粉丝数量超1200万。他曾被称为“中老年...
9月3日消息,亚马逊的一些股东,包括持有该公司股票的一家养老基金,日前对亚马逊、其创始人贝索斯和其董事会提起诉讼,指控他们在为 Project Kuiper 卫星星座项目购买发射服务时“违反了信义义务”。
据消息,为推广自家应用,苹果现推出了一个名为“Apps by Apple”的网站,展示了苹果为旗下产品(如 iPhone、iPad、Apple Watch、Mac 和 Apple TV)开发的各种应用程序。
特斯拉本周在美国大幅下调Model S和X售价,引发了该公司一些最坚定支持者的不满。知名特斯拉多头、未来基金(Future Fund)管理合伙人加里·布莱克发帖称,降价是一种“短期麻醉剂”,会让潜在客户等待进一步降价。
据外媒9月2日报道,荷兰半导体设备制造商阿斯麦称,尽管荷兰政府颁布的半导体设备出口管制新规9月正式生效,但该公司已获得在2023年底以前向中国运送受限制芯片制造机器的许可。
近日,根据美国证券交易委员会的文件显示,苹果卫星服务提供商 Globalstar 近期向马斯克旗下的 SpaceX 支付 6400 万美元(约 4.65 亿元人民币)。用于在 2023-2025 年期间,发射卫星,进一步扩展苹果 iPhone 系列的 SOS 卫星服务。
据报道,马斯克旗下社交平台𝕏(推特)日前调整了隐私政策,允许 𝕏 使用用户发布的信息来训练其人工智能(AI)模型。新的隐私政策将于 9 月 29 日生效。新政策规定,𝕏可能会使用所收集到的平台信息和公开可用的信息,来帮助训练 𝕏 的机器学习或人工智能模型。
9月2日,荣耀CEO赵明在采访中谈及华为手机回归时表示,替老同事们高兴,觉得手机行业,由于华为的回归,让竞争充满了更多的可能性和更多的魅力,对行业来说也是件好事。
《自然》30日发表的一篇论文报道了一个名为Swift的人工智能(AI)系统,该系统驾驶无人机的能力可在真实世界中一对一冠军赛里战胜人类对手。
近日,非营利组织纽约真菌学会(NYMS)发出警告,表示亚马逊为代表的电商平台上,充斥着各种AI生成的蘑菇觅食科普书籍,其中存在诸多错误。
社交媒体平台𝕏(原推特)新隐私政策提到:“在您同意的情况下,我们可能出于安全、安保和身份识别目的收集和使用您的生物识别信息。”
2023年德国柏林消费电子展上,各大企业都带来了最新的理念和产品,而高端化、本土化的中国产品正在不断吸引欧洲等国际市场的目光。
罗永浩日前在直播中吐槽苹果即将推出的 iPhone 新品,具体内容为:“以我对我‘子公司’的了解,我认为 iPhone 15 跟 iPhone 14 不会有什么区别的,除了序(列)号变了,这个‘不要脸’的东西,这个‘臭厨子’。