TypeScript 之 Typeof Type Operator
TypeScript 的官方文档早已更新,但我能找到的中文文档都还停留在比较老的版本。所以对其中新增以及修订较多的一些章节进行了翻译整理。本篇整理自 TypeScript Handbook 中 「Typeof Type Operator」 章节。本文并不严格按照原文翻译,对部分内容也做了解释补充。
typeof
类型操作符(The typeof
type operator)
JavaScript 本身就有 typeof
操作符,你可以在表达式上下文中(expression context)使用:
// Prints "string" console.log(typeof "Hello world");
而 TypeScript 添加的 typeof
方法可以在类型上下文(type context)中使用,用于获取一个变量或者属性的类型。
let s = "hello"; let n: typeof s; // let n: string
如果仅仅用来判断基本的类型,自然是没什么太大用,和其他的类型操作符搭配使用才能发挥它的作用。
举个例子:比如搭配 TypeScript 内置的 ReturnType<T>
。你传入一个函数类型,ReturnType<T>
会返回该函数的返回值的类型:
type Predicate = (x: unknown) => boolean; type K = ReturnType<Predicate>; /// type K = boolean
如果我们直接对一个函数名使用 ReturnType
,我们会看到这样一个报错:
function f() { return { x: 10, y: 3 }; } type P = ReturnType<f>; // 'f' refers to a value, but is being used as a type here. Did you mean 'typeof f'?
这是因为值(values)和类型(types)并不是一种东西。为了获取值 f
也就是函数 f
的类型,我们就需要使用 typeof
:
function f() { return { x: 10, y: 3 }; } type P = ReturnType<typeof f>; // type P = { // x: number; // y: number; // }
限制(Limitations)
TypeScript 有意的限制了可以使用 typeof
的表达式的种类。
在 TypeScript 中,只有对标识符(比如变量名)或者他们的属性使用 typeof
才是合法的。这可能会导致一些令人迷惑的问题:
// Meant to use = ReturnType<typeof msgbox> let shouldContinue: typeof msgbox("Are you sure you want to continue?"); // ',' expected.
我们本意是想获取 msgbox("Are you sure you want to continue?")
的返回值的类型,所以直接使用了 typeof msgbox("Are you sure you want to continue?")
,看似能正常执行,但实际并不会,这是因为 typeof
只能对标识符和属性使用。而正确的写法应该是:
ReturnType<typeof msgbox>
(注:原文到这里就结束了)
对对象使用 typeof
我们可以对一个对象使用 typeof
:
const person = { name: "kevin", age: "18" } type Kevin = typeof person; // type Kevin = { // name: string; // age: string; // }
对函数使用 typeof
我们也可以对一个函数使用 typeof
:
function identity<Type>(arg: Type): Type { return arg; } type result = typeof identity; // type result = <Type>(arg: Type) => Type
对 enum 使用 typeof
在 TypeScript 中,enum 是一种新的数据类型,但在具体运行的时候,它会被编译成对象。
enum UserResponse { No = 0, Yes = 1, }
对应编译的 JavaScript 代码为:
var UserResponse; (function (UserResponse) { UserResponse[UserResponse["No"] = 0] = "No"; UserResponse[UserResponse["Yes"] = 1] = "Yes"; })(UserResponse || (UserResponse = {}));
如果我们打印一下 UserResponse
:
console.log(UserResponse); // [LOG]: { // "0": "No", // "1": "Yes", // "No": 0, // "Yes": 1 // }
而如果我们对 UserResponse
使用 typeof
:
type result = typeof UserResponse; // ok const a: result = { "No": 2, "Yes": 3 } result 类型类似于: // { // "No": number, // "YES": number // }
不过对一个 enum 类型只使用 typeof
一般没什么用,通常还会搭配 keyof
操作符用于获取属性名的联合字符串:
type result = keyof typeof UserResponse; // type result = "No" | "Yes"
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(7)
这个题目想考察哪个知识点
大佬问下这个问题
enum UserResponse {
No = '0',
Yes = '1',
}
// 如果是数字枚举result 类型类似于:
// {
// "No": number,
// "YES": number
// }
// 如果是字符串枚举应该会是什么样呢?
let a = UserResponse.Yes
type result = typeof a;
let de: result = '1' // 为什么不能赋值?
@zhijunw 感谢~ 主要是原文的文档写的好,哈哈
@NameWjp 感谢勘误,已更正~
写的好仔细 每天学习一篇
写的好仔细 每天学习一篇
举个例子:比如搭配 TypeScript 内置的 ReturnTypep。你传入一个函数类型,ReturnTypep 会返回该函数的返回值的类型
=>
举个例子:比如搭配 TypeScript 内置的 ReturnType。你传入一个函数类型,ReturnType 会返回该函数的返回值的类型: