My Personal Blog

Published on

第三十章 TypeScript 类型运算

Authors
  • avatar
    Name
    Tian Haipeng

运算律

改变成员类型的顺序不影响联合类型的结果类型。

type T0 = string | number
type T1 = number | string

对部分类型成员使用分组运算符不影响联合类型的结果类型。

type T0 = (boolean | string) | number
type T1 = boolean | (string | number)

联合类型的成员类型可以进行化简。假设有联合类型“U = T0 | T1”,如果 T1 是 T0 的子类型,那么可以将类型成员 T1 从联合类型 U 中消去。最后,联合类型 U 的结果类型为“U = T0”。例如,有联合类型“boolean | true | false”。其中,true 类型和 false 类型是 boolean 类型的子类型,因此可以将 true 类型和 false 类型从联合类型中消去。最终,联合类型“boolean | true | false”的结果类型为 boolean 类型。

type T0 = boolean | true | false

// 所以T0等同于 T1
type T1 = boolean

优先级

&的优先级高于|

;(A & B) |
  (C &
    D(
      // 该类型等同于如下类型:
      A & B
    )) |
  (C & D)

分配律

;(A &
  (B | C)(
    // 等同于
    A & B
  )) |
  (A & C)

一个稍微复杂的类型等式。

(A | B) & (C | D)A & C | A & D | B & C | B & D
T = (string | 0) & (number | 'a')
T = (string & number) | (string & 'a') | (0 & number) | (0 & 'a')

T = never | 'a' | 0 | never
T = 'a' | 0
function extend<T extends object, U extends object>(first: T, second: U): T & U {
  const result = <T & U>{}
  for (let id in first) {
    ;(<T>result)[id] = first[id]
  }
  for (let id in second) {
    if (!result.hasOwnProperty(id)) {
      ;(<U>result)[id] = second[id]
    }
  }

  return result
}

const x = extend({ a: 'hello' }, { b: 42 })

never 类型

never 可以视为空集。

type NeverIntersection = never & string // Type: never
type NeverUnion = never | string // Type: string

很适合在交叉类型中用作过滤。

type OnlyStrings<T> = T extends string ? T : never
type RedOrBlue = OnlyStrings<'red' | 'blue' | 0 | false>
// Equivalent to: "red" | "blue"

范例:https://www.typescriptlang.org/play#example/conditional-types

unknown 类型

在联合类型中,unknown 吸收所有类型。这意味着如果任何组成类型是 unknown,则联合类型的计算结果为 unknown。

// In an intersection everything absorbs unknown
type T00 = unknown & null // null
type T01 = unknown & undefined // undefined
type T02 = unknown & null & undefined // null & undefined (which becomes never)
type T03 = unknown & string // string
type T04 = unknown & string[] // string[]
type T05 = unknown & unknown // unknown
type T06 = unknown & any // any
// In a union an unknown absorbs everything
type T10 = unknown | null // unknown
type T11 = unknown | undefined // unknown
type T12 = unknown | null | undefined // unknown
type T13 = unknown | string // unknown
type T14 = unknown | string[] // unknown
type T15 = unknown | unknown // unknown
type T16 = unknown | any // any
// Type variable and unknown in union and intersection
type T20<T> = T & {} // T & {}
type T21<T> = T | {} // T | {}
type T22<T> = T & unknown // T
type T23<T> = T | unknown // unknown
// unknown in conditional types
type T30<T> = unknown extends T ? true : false // Deferred
type T31<T> = T extends unknown ? true : false // Deferred (so it distributes)
type T32<T> = never extends T ? true : false // true
type T33<T> = T extends never ? true : false // Deferred
type UnionType1 = unknown | null // unknown
type UnionType2 = unknown | undefined // unknown
type UnionType3 = unknown | string // unknown
type UnionType4 = unknown | number[] // unknown

该规则的一个例外是 any。如果至少有一种构成类型是 any,则联合类型的计算结果为 any:

type UnionType5 = unknown | any // any

在交叉类型中,每种类型都吸收 unknown. 这意味着与任何类型相交 unknown 不会改变结果类型:

type IntersectionType1 = unknown & null // null
type IntersectionType2 = unknown & undefined // undefined
type IntersectionType3 = unknown & string // string
type IntersectionType4 = unknown & number[] // number[]
type IntersectionType5 = unknown & any // any

除非使用as断言,首先缩小类型unknown类型的范围,然后才可以用于其他类型。

const value: unknown = 'Hello World'
const someString: string = value as string
const otherString = someString.toUpperCase() // "HELLO WORLD"

联合类型

如果类型是多个值的联合,甚至可以产生插值的效果。

type EmailLocaleIDs = 'welcome_email' | 'email_heading'
type FooterLocaleIDs = 'footer_title' | 'footer_sendoff'

// 等同于 type AllLocaleIDs = "welcome_email_id" | "email_heading_id" | "footer_title_id" | "footer_sendoff_id"
type AllLocaleIDs = `${EmailLocaleIDs | FooterLocaleIDs}_id`

交叉类型

type Brightness = 'dark' | 'light'
type Color = 'blue' | 'red'
type BrightnessAndColor = `${Brightness}-${Color}`
// Equivalent to: "dark-red" | "light-red" | "dark-blue" | "light-blue"

如果交叉类型中存在多个相同的成员类型,那么相同的成员类型将被合并为单一成员类型。

type T0 = boolean
type T1 = boolean & boolean
type T2 = boolean & boolean & boolean

上面示例中,T0、T1 和 T2 都表示同一种类型 boolean。

改变成员类型的顺序不影响交叉类型的结果类型。

interface Clickable {
  click(): void
}
interface Focusable {
  focus(): void
}

type T0 = Clickable & Focusable
type T1 = Focusable & Clickable

注意,当交叉类型涉及调用签名重载或构造签名重载时便失去了“加法交换律”的性质。因为交叉类型中成员类型的顺序将决定重载签名的顺序,进而将影响重载签名的解析顺序。

interface Clickable {
  register(x: any): void
}
interface Focusable {
  register(x: string): boolean
}

type ClickableAndFocusable = Clickable & Focusable
type FocusableAndFocusable = Focusable & Clickable

function foo(clickFocus: ClickableAndFocusable, focusClick: FocusableAndFocusable) {
  let a: void = clickFocus.register('foo')
  let b: boolean = focusClick.register('foo')
}

此例第 8 行和第 9 行使用不同的成员类型顺序定义了两个交叉类型。第 15 行,调用“register()”方法的返回值类型为 void,说明在 ClickableAndFocusable 类型中,Clickable 接口中定义的“register()”方法具有更高的优先级。第 16 行,调用“register()”方法的返回值类型为 boolean,说明 FocusableAndFocusable 类型中 Focusable 接口中定义的“register()”方法具有更高的优先级。此例也说明了调用签名重载的顺序与交叉类型中成员类型的定义顺序是一致的。

对部分类型成员使用分组运算符不影响交叉类型的结果类型。

interface Clickable {
  click(): void
}
interface Focusable {
  focus(): void
}
interface Scrollable {
  scroll(): void
}

type T0 = (Clickable & Focusable) & Scrollable
type T1 = Clickable & (Focusable & Scrollable)

上面示例的 T0 和 T1 类型是同一种类型。

type Combined = { a: number } & { b: string }
type Conflicting = { a: number } & { a: string }

只要交叉类型 I 中任意一个成员类型包含了属性签名 M,那么交叉类型 I 也包含属性签名 M。

interface A {
  a: boolean
}

interface B {
  b: string
}

// 交叉类型如下
{
  a: boolean
  b: string
}

若交叉类型的属性签名 M 在所有成员类型中都是可选属性,那么该属性签名在交叉类型中也是可选属性。否则,属性签名 M 是一个必选属性。

interface A {
    x: boolean;
    y?: string;
}
interface B {
    x?: boolean;
    y?: string;
}

// 交叉类型如下
{
    x: boolean;
    y?: string;
}