夜猫子的知识栈 夜猫子的知识栈
首页
  • 前端文章

    • JavaScript
  • 学习笔记

    • 《JavaScript教程》
    • 《Web Api》
    • 《ES6教程》
    • 《Vue》
    • 《React》
    • 《TypeScript》
    • 《Git》
    • 《Uniapp》
    • 小程序笔记
    • 《Electron》
    • JS设计模式总结
  • 《前端架构》

    • 《微前端》
    • 《权限控制》
    • monorepo
  • 全栈项目

    • 任务管理日历
    • 无代码平台
    • 图书管理系统
  • HTML
  • CSS
  • Nodejs
  • Midway
  • Nest
  • MySql
  • 其他
  • 技术文档
  • GitHub技巧
  • 博客搭建
  • Ajax
  • Vite
  • Vitest
  • Nuxt
  • UI库文章
  • Docker
  • 学习
  • 面试
  • 心情杂货
  • 实用技巧
  • 友情链接
收藏
  • 分类
  • 标签
  • 归档
GitHub (opens new window)

夜猫子

前端练习生
首页
  • 前端文章

    • JavaScript
  • 学习笔记

    • 《JavaScript教程》
    • 《Web Api》
    • 《ES6教程》
    • 《Vue》
    • 《React》
    • 《TypeScript》
    • 《Git》
    • 《Uniapp》
    • 小程序笔记
    • 《Electron》
    • JS设计模式总结
  • 《前端架构》

    • 《微前端》
    • 《权限控制》
    • monorepo
  • 全栈项目

    • 任务管理日历
    • 无代码平台
    • 图书管理系统
  • HTML
  • CSS
  • Nodejs
  • Midway
  • Nest
  • MySql
  • 其他
  • 技术文档
  • GitHub技巧
  • 博客搭建
  • Ajax
  • Vite
  • Vitest
  • Nuxt
  • UI库文章
  • Docker
  • 学习
  • 面试
  • 心情杂货
  • 实用技巧
  • 友情链接
收藏
  • 分类
  • 标签
  • 归档
GitHub (opens new window)
  • TypeScript基础

    • TypeScript笔记
    • tsconfig.json配置
    • Record对象键值定义
    • 对象的方式描述类型
    • implements和extends
    • interface和type
    • keyof和typeof
    • Partial详解
      • Partial 类型的定义
      • keyof
      • in
      • T[P]
    • 常用泛型工具
  • TypeScript进阶

  • TS 从零实现 axios

  • 《TypeScript》
  • TypeScript基础
夜猫子
2022-11-03
目录

Partial详解

# TypeScript高级类型-Partial

预备知识:

TypeScript类型系统 接口 泛型

# Partial 类型的定义

/**
 * Make all properties in T optional
 */
 type Partial<T> = {
   [P in keyof T]?: T[P];
  };

1
2
3
4
5
6
7

假设我们有一个定义 user 的接口,如下

interface IUser {
  name: string
  age: number
  department: string
}
1
2
3
4
5

经过 Partial 类型转化后得到

type optional = Partial<IUser>

// optional的结果如下
type optional = {
    name?: string | undefined;
    age?: number | undefined;
    department?: string | undefined;
}
1
2
3
4
5
6
7
8

那么 Partial<T> 是如何实现类型转化的呢?

1.遍历入参 T ,获得每一对 key, value

2.将每一对中的 key 变为可选,即添加 ?

3.希望得到的是由 key, value 组成的新类型

以上对应到 TypeScript 中是如何实现的呢?

对照最开始 Partial 的类型定义,能够捕捉到以下重要信息

  • keyof 是干什么的?

  • in 是干什么的?

  • [P in keyof T] 中括号是干什么的?

  • ? 是将该属性变为可选属性(此外,-?表示必选,比如 Required 类型)

  • T[P] 是干什么的?

# keyof

keyof,即 索引类型查询操作符,我们可以将 keyof 作用于泛型 T 上来获取泛型 T 上的所有 public 属性名构成的 联合类型

注意:"public、protected、private"修饰符不可出现在类型成员上

例如:

type unionKey = keyof IUser

// unionKey 结果如下,其获得了接口类型 IUser 中的所有属性名组成的联合类型
type unionKey = "name" | "age" | "department"
1
2
3
4

# in

我们需要遍历 IUser ,这时候 映射类型就可以用上了,其语法为 [P in Keys]

  • P:类型变量,依次绑定到每个属性上,对应每个属性名的类型

  • Keys:字符串字面量构成的联合类型,表示一组属性名(的类型),可以联想到上文 keyof 的作用

上述问题中 [P in keyof T] 中括号是干什么的?这里也就很清晰了。

# T[P]

我们可以通过 keyof 查询索引类型的属性名,那么如何获取属性名对应的属性值类型呢?

这里就用到了 索引访问操作符,与 JavaScript 种访问属性值的操作类似,访问类型的操作符也是通过 [] 来访问的,即 T[P],其中”中括号“中的 P 与 [P in keyof T] 中的 P 相对应。

例如

type unionKey = keyof IUser // "name" | "age" | "department"

type values = IUser[unionKey] // string | number 属性值类型组成的联合类型
1
2
3

最后我们希望得到的是由多个 key, value 组成的新类型,故而在 [P in keyof T]?: T[P]; 外部包上一层大括号。

到此我们解决遇到的所有问题,只需要逐步代入到 Partial 类型定义中即可。

编辑 (opens new window)
上次更新: 2023/9/13 18:19:54
keyof和typeof
常用泛型工具

← keyof和typeof 常用泛型工具→

最近更新
01
IoC 解决了什么痛点问题?
03-10
02
如何调试 Nest 项目
03-10
03
Provider注入对象
03-10
更多文章>
Copyright © 2019-2025 Study | MIT License
  • 跟随系统
  • 浅色模式
  • 深色模式
  • 阅读模式