一、什么是「映射类型」?
一句话定义:
用一个已有类型的 key,批量生成一个新类型
核心语法:
ts
type NewType<T> = {
[K in keyof T]: T[K];
};📌 这里的 in 不是 JS 的 in,而是 类型层面的遍历 key
二、最基础示例(一定要会)
1️⃣ 手写 Readonly<T>
ts
type MyReadonly<T> = {
readonly [K in keyof T]: T[K];
};等价于 TS 内置的:
ts
Readonly<T>;2️⃣ 手写 Partial<T>
ts
type MyPartial<T> = {
[K in keyof T]?: T[K];
};3️⃣ 手写 Required<T>
ts
type MyRequired<T> = {
[K in keyof T]-?: T[K];
};📌 -? 表示 移除可选修饰符
三、映射类型中的「修饰符」
修饰符一览
| 修饰符 | 含义 |
|---|---|
readonly | 只读 |
? | 可选 |
-readonly | 移除只读 |
-? | 移除可选 |
示例:去掉 readonly
ts
type Mutable<T> = {
-readonly [K in keyof T]: T[K];
};四、映射 + 条件判断(进阶)
1️⃣ 根据 key 过滤字段(核心技巧)
ts
type FilterString<T> = {
[K in keyof T as T[K] extends string ? K : never]: T[K];
};ts
type User = {
id: number;
name: string;
age: number;
};
type StringProps = FilterString<User>;
// { name: string }📌 as 在映射类型里可以改 key
2️⃣ key 重命名(Vue / React 源码常见)
ts
type Getters<T> = {
[K in keyof T as `get${Capitalize<string & K>}`]: () => T[K];
};ts
type State = {
count: number;
name: string;
};结果:
ts
{
getCount: () => number;
getName: () => string;
}五、in 的本质理解(非常重要)
ts
[K in keyof T]等价于:
ts
K = "a" | "b" | "c";👉 TS 会对 联合类型自动展开
ts
type Keys = "a" | "b";
type Obj = {
[K in Keys]: number;
};
// { a: number; b: number }六、真实前端场景
1️⃣ 表单校验规则
ts
type Form = {
username: string;
password: string;
};
type Rules<T> = {
[K in keyof T]: {
required?: boolean;
message?: string;
};
};2️⃣ Vue props 默认值
ts
type Props = {
size: string;
disabled: boolean;
};
type DefaultProps = {
[K in keyof Props]?: Props[K];
};3️⃣ API 状态映射
ts
type Api = {
getUser: () => void;
getList: () => void;
};
type LoadingState<T> = {
[K in keyof T]: boolean;
};七、常见误区(面试高频)
❌ in 只能用在类型里
ts
[K in arr]; // ❌❌ 忘记 keyof
ts
[K in T] // ❌
[K in keyof T] // ✅❌ 以为 in 是运行时
ts
// 映射类型不会生成 JS 代码八、映射类型 = TS 工具类型的底层
| 工具类型 | 底层 |
|---|---|
Partial | in + ? |
Required | in + -? |
Readonly | in + readonly |
Pick | in + 条件 |
Record | in |
九、记忆公式 🧠
- keyof:拿 key
- in:遍历 key
- as:改 key
- extends:筛 key