这篇文章就来分享一些很多人不知道的小技巧吧,都是学完就能用起来的那种。
创新互联公司是一家企业级云计算解决方案提供商,超15年IDC数据中心运营经验。主营GPU显卡服务器,站群服务器,服务器托管,海外高防服务器,机柜大带宽、租用·托管,动态拨号VPS,海外云手机,海外云服务器,海外服务器租用托管等。
TypeScript 有一个内置类型叫做 Record,它的作用是根据传入的索引和值的类型构造新的索引类型。
它的实现就是通过映射类型的语法构造一个索引类型:
type Record= { [P in K]: T };
那么问题来了,这个 K 怎么约束呢?
有同学说 K 不是索引么?那应该是 string,也就是 K extends string。
但是 JS 的属性可以是 string、number、symbol 这三种类型的。
那我知道了,要 K extends string | number | symbol。
不不不,TypeScript 有个编译选项叫做 keyofStringsOnly,开启了那么就就只会用 string 作为索引,否则才是 string | number | symbol:
这还与编译选项有关,那这里改怎么约束呢?
看下 TS 源码里是怎么定义 Record 的:
type Record= { [P in K]: T; };
它用了 keyof any,难道这个 keyof any 就能动态得到 key 支持的类型么?
我们试一下,不开启 keyofStringsOnly 时:
开启 keyofStringsOnly 时:
妙啊,这样就能动态获取当前支持的 key 的类型了。
需要约束某个类型参数为索引 Key 时,用 keyof any 动态获取比写死 string | number | symbol 更好。
object 和 RecordTypeScript 里有三个类型比较难区分,就是 object、Object、{} 这几个。
其实只要记住 object 不能接受原始类型 就可以了,其余两个差不多,只不过 {} 是个空对象,没有索引。
所以 number 就可以赋值给 {}、Object 类型,但是不能赋值给 object 类型:
其实,你看源码会发现大家不会用 object 来约束,而是用 Record 来约束索引类型,这俩其实是一样的,但是 Record 更语义化一些。
Record 创建了一个 key 为任意 string,value 为任意类型的索引类型:
所以,平时约束索引类型的时候就可以用 Record 代替 object。
而且你会在很多源码里看到这种写法,比如下面是 Nest.js 源码里的:
映射类型可以构造一个新的索引类型,并且构造的过程中做一些修改。
比如构造一个新的索引类型,把所有的 Key 变为可选:
type ToPartial= { [Key in keyof T]?: T[Key] }
或者构造一个新的索引类型,加上 readonly 的修饰:
type ToReadonly = { readonly [Key in keyof T]: T[Key]; }
但很多人不知道也可以去掉已有的修饰的,用 - 号,减去的意思:
比如去掉 ? 是 -? :
type ToRequired= { [Key in keyof T]-?: T[Key] }
那去掉 readonly 自然就是 -readonly:
type ToMutable= { -readonly [Key in keyof T]: T[Key] }
我最近看到 Promise.all 的类型定义就用到这个了:
类型参数 T 是 待处理的 promise 数组,返回值是 Promise 的 value 对应的数组,用 Awaited 取出 value 的类型。
Awaited 是 TS 内置的一个高级类型,用于取出 Promise 返回值类型的:
返回的是数组类型,那为啥还可以用映射类型的语法呢?
因为数组类型也是索引类型呀,索引类型的意思就是聚合多个元素的类型,数组、对象、class 都是索引类型。
当然,主要还是为了讲 -readonly 的语法,可以去掉 readonly 的修饰。
方法里可以调用 this,比如这样:
class Dong {
name: string;
constructor() {
this.name = "dong";
}
hello() {
return 'hello, I\'m ' + this.name;
}
}
const dong = new Dong();
dong.hello();
用对象.方法名的方式调用的时候,this 就指向那个对象。
但是方法也可以用 call 或者 apply 调用:
call 调用的时候,this 就变了,但这里却没有被检查出来 this 指向的错误。
如何让编译器能够检查出 this 指向的错误呢?
其实方法是可以指定 this 的类型的:
class Dong {
name: string;
constructor() {
this.name = "dong";
}
hello(this: Dong) {
return 'hello, I\'m ' + this.name;
}
}
这样,当 call/apply 调用的时候,就能检查出 this 指向的对象是否是对的:
而且,TypeScript 也提供了一个内置的高级类型 ThisParameterType 用于提取 this 的类型:
它的实现很简单,就是通过模式匹配提取 this 的类型到 infer 声明的局部变量里返回:
最后是一个比较常用的语法,TS 支持 ? 的可选链语法,也可以通过 ?? 指定默认值:
const dong = data?.name ?? 'dong';
编译之后会变成这样:
做了空值检查,也设置了默认值 dong。
很简单和有用的一个语法,但很多人写 ts 还是没把它用起来。
TypeScript 有很多灵活的语法,小技巧很多。
今天分享了一些大家可能不知道的技巧:
这几个小技巧都是看一遍就会的那种,下次写 TS 类型的时候就可以用起来了。
网站题目:几个一看就会的TypeScript小技巧
当前路径:http://www.csdahua.cn/qtweb/news21/260771.html
网站建设、网络推广公司-快上网,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等
声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 快上网