TypeScript骚操作

2023/11/27

可变参数

写过ReactNative的都知道,ReactNavigation库里面有一个非常牛逼的类型声明,它可以根据你传入的参数,来判断是否需要第二个参数。

例如下面的定义:

interface RouteDef { home: undefined shop: { time: number } }
typescript

那么它的路由方法就变得很牛逼了:

// 实际的泛型肯定不是这么传的,因为我写的时候没用RN了,所有忘了咋用的了 const nav = useNavigation<RouteDef>() // 正确 nav('home') // 错误: ts提示这里只需要一个参数 nav('home', undefined) // 错误,ts提示这里需要两个参数 nav('shop') // 正确 nav('shop', { time: Date.now() }) // 错误: 第二个参数需要{ time: number }类型,而提供的是number nav('shop', Date.now())
typescript

很牛逼啊有没有,连参数的数量都给你变了!而且还能根据属性的定义来决定参数。

那么这玩意具体是咋写的呢?我仔细研究了一下,研究完后,仿佛开启了新大门!

我这里直接写一个Demo来看:

type EmitFuncArgs<Events, Key extends keyof Events> = void extends Events[Key] ? [evt: Key] : [evt: Key, data: Events[Key]] type EmitFunc<Events> = <T extends keyof Events> (...args: EmitFuncArgs<Events, T>) => string export interface AppEvents { ON_LOGOUT: string ON_LOGIN: void }
typescript

先来看下面的EmitFunc定义,首先这玩意的泛型接收一个Events类型(直接把它看成下面的AppEvents接口就行),然后它的参数是EmitFuncArgs 决定的。

来看EmitFuncArgs,可以发现这玩意居然返回了一个数组,并且还用上了三目运算符。

到了这里,大伙都应该可以理解是怎么玩的了,就是首先判断值是不是void,如果是,则返回一个长度为1的数组,反之则返回长度为2的数组,其中的第二个 参数为接口定义的类型。