TypeScript 中的"空对象字面值"
3 min
在做 Type Challenges 时,
对 AnyOf
这道迷惑了很久:
我最开始的题解如下,将 falsy 的类型统一成一个 tuple,再判断每个元素是否 truthy。
type FalsyType = '' | [] | false | 0 | {}
type AnyOf<T extends readonly any[]> = T extends [infer L, ...infer R]
? L extends FalsyType
? AnyOf<R>
: true
: false
当然,这种写法是错误的。
因为空对象字面值 {}
是一个 falsy 的类型,但 T extends {}
判断的却是 T
是否是一个对象,
对于 JavaScript 这样万物皆对象的语言这种条件判断便显得毫无意义了。
于是在我的题解中,利用 Index Signature 针对 {}
进行了单独的判断,而不是将它与其他 falsy 类型统一到一个 tuple 中。
type AnyOf<T extends readonly any[]> = T extends [infer L, ...infer R]
? L extends '' | [] | false | 0
? AnyOf<R>
: keyof L extends undefined[]
? AnyOf<R>
: true
: false
今天,偶然在另一个项目的 @typescript-eslint
提示中看到这样的信息:
Don't use `{}` as a type. `{}` actually means "any non-nullish value".
- If you want a type meaning "any object", you probably want `Record<string, unknown>` instead.
- If you want a type meaning "any value", you probably want `unknown` instead.
- If you want a type meaning "empty object", you probably want `Record<string, never>` instead.
即:
不要将 {}
用作类型,因为 {}
实际上表示”任何非空值”。
- 如果要表示任何对象,用
Record<string, unknown>
。 - 如果要表示任何值,用
unknown
。 - 如果要表示空对象字面值,用
Record<string, never>
。
据此,就可以将题解改写为如下形式了:
type FalsyType = '' | [] | false | 0 | Record<string, never>
type AnyOf<T extends readonly any[]> = T extends [infer L, ...infer R]
? L extends FalsyType
? AnyOf<R>
: true
: false
最后吐槽一下,当时在 bing、google、stackoverflow 上搜了很久 TS 中空字符字面量怎么表示,但一直没查到, 大概是英语水平不行吧,单凭敲几个关键词还是不能准确检索。