欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

TypeScript中條件類型精讀與實(shí)踐記錄

 更新時(shí)間:2021年10月05日 08:25:10   作者:牧云云  
這篇文章主要給大家介紹了關(guān)于TypeScript中條件類型精讀與實(shí)踐的相關(guān)資料,,條件類型就是在初始狀態(tài)并不直接確定具體類型,而是通過(guò)一定的類型運(yùn)算得到最終的變量類型,需要的朋友可以參考下

在大多數(shù)程序中,我們必須根據(jù)輸入做出決策。TypeScript 也不例外,使用條件類型可以描述輸入類型與輸出類型之間的關(guān)系。

用于條件判斷時(shí)的 extends

當(dāng) extends 用于表示條件判斷時(shí),可以總結(jié)出以下規(guī)律

若位于 extends 兩側(cè)的類型相同,則 extends 在語(yǔ)義上可理解為 ===,可以參考如下例子:

type result1 = 'a' extends 'abc' ? true : false // false
type result2 = 123 extends 1 ? true : false     // false

若位于 extends 右側(cè)的類型包含位于 extends 左側(cè)的類型(即狹窄類型 extends 寬泛類型)時(shí),結(jié)果為 true,反之為 false??梢詤⒖既缦吕?

type result3 = string extends string | number ? true : false // true

當(dāng) extends 作用于對(duì)象時(shí),若在對(duì)象中指定的 key 越多,則其類型定義的范圍越狹窄。可以參考如下例子:

type result4 = { a: true, b: false } extends { a: true } ? true : false // true

在泛型類型中使用條件類型

考慮如下 Demo 類型定義:

type Demo<T, U> = T extends U ? never : T

結(jié)合用于條件判斷時(shí)的 extends,可知 'a' | 'b' | 'c' extends 'a' 是 false, 因此 Demo<'a' | 'b' | 'c', 'a'> 結(jié)果是 'a' | 'b' | 'c' 么?
查閱官網(wǎng),其中有提到:

When conditional types act on a generic type, they become distributive when given a union type.

即當(dāng)條件類型作用于泛型類型時(shí),聯(lián)合類型會(huì)被拆分使用。即 Demo<'a' | 'b' | 'c', 'a'> 會(huì)被拆分為 'a' extends 'a'、'b' extends 'a'、'c' extends 'a'。用偽代碼表示類似于:

function Demo(T, U) {
  return T.map(val => {
    if (val !== U) return val
    return 'never'
  })
}

Demo(['a', 'b', 'c'], 'a') // ['never', 'b', 'c']

此外根據(jù) never 類型的定義 —— never 類型可分配給每種類型,但是沒(méi)有類型可以分配給 never(除了 never 本身)。即 never | 'b' | 'c' 等價(jià)于 'b' | 'c'。

因此 Demo<'a' | 'b' | 'c', 'a'> 的結(jié)果并不是 'a' | 'b' | 'c' 而是 'b' | 'c'。

工具類型

心細(xì)的讀者可能已經(jīng)發(fā)現(xiàn)了 Demo 類型的聲明過(guò)程其實(shí)就是 TypeScript 官方提供的工具類型中 Exclude<Type, ExcludedUnion> 的實(shí)現(xiàn)原理,其用于將聯(lián)合類型 ExcludedUnion 排除在 Type 類型之外。

type T = Demo<'a' | 'b' | 'c', 'a'> // T: 'b' | 'c'

基于 Demo 類型定義,進(jìn)一步地還可以實(shí)現(xiàn)官方工具類型中的 Omit<Type, Keys>,其用于移除對(duì)象 Type
中滿足 keys 類型的屬性值。

type Omit<Type, Keys> = {
  [P in Demo<keyof Type, Keys>]: Type<P>
}

interface Todo {
  title: string;
  description: string;
  completed: boolean;
}

type T = Omit<Todo, 'description'> // T: { title: string; completed: boolean }

逃離艙

如果想讓 Demo<'a' | 'b' | 'c', 'a'> 的結(jié)果為 'a' | 'b' | 'c' 是否可以實(shí)現(xiàn)呢? 根據(jù)官網(wǎng)描述:

Typically, distributivity is the desired behavior. To avoid that behavior, you can surround each side of the extends keyword with square brackets.

如果不想遍歷泛型中的每一個(gè)類型,可以用方括號(hào)將泛型給括起來(lái)以表示使用該泛型的整體部分。
type Demo<T, U> = [T] extends [U] ? never : T

type Demo<T, U> = [T] extends [U] ? never : T

// result 此時(shí)類型為 'a' | 'b' | 'c'
type result = Demo<'a' | 'b' | 'c', 'a'>

在箭頭函數(shù)中使用條件類型

在箭頭函數(shù)中使用三元表達(dá)式時(shí),從左向右的閱讀習(xí)慣導(dǎo)致函數(shù)內(nèi)容區(qū)若不加括號(hào)則會(huì)讓使用方感到困惑。比如下方代碼中 x 是函數(shù)類型還是布爾類型呢?

// The intent is not clear.
var x = a => 1 ? true : false

在 eslint 規(guī)則 no-confusing-arrow 中,推薦如下寫(xiě)法:

var x = a => (1 ? true : false)

在 TypeScript 的類型定義中,若在箭頭函數(shù)中使用 extends 也是同理,由于從左向右的閱讀習(xí)慣,也會(huì)導(dǎo)致閱讀者對(duì)類型代碼的執(zhí)行順序感到困惑。

type Curry<P extends any[], R> =
  (arg: Head<P>) => HasTail<P> extends true ? Curry<Tail<P>, R> : R

因此在箭頭函數(shù)中使用 extends 建議加上括號(hào),對(duì)于進(jìn)行 code review 有很大的幫助。

type Curry<P extends any[], R> =
  (arg: Head<P>) => (HasTail<P> extends true ? Curry<Tail<P>, R> : R)

結(jié)合類型推導(dǎo)使用條件類型

在 TypeScript 中,一般會(huì)結(jié)合 extends 來(lái)使用類型推導(dǎo) infer 語(yǔ)法。使用它可以實(shí)現(xiàn)自動(dòng)推導(dǎo)類型的目的。比如用其來(lái)實(shí)現(xiàn)工具類型 ReturnType<Type> ,該工具類型用于返回函數(shù) Type 的返回類型。
type ReturnType<T extends Function> = T extends (...args: any) => infer U ? U : never

type ReturnType<T extends Function> = T extends (...args: any) => infer U ? U : never

MyReturnType<() => string>          // string
MyReturnType<() => Promise<boolean> // Promise<boolean>

結(jié)合 extends 與類型推導(dǎo)還可以實(shí)現(xiàn)與數(shù)組相關(guān)的 Pop<T>、Shift<T>、Reverse<T> 工具類型。

Pop<T>:

type Pop<T extends any[]> = T extends [...infer ExceptLast, any] ? ExceptLast : never

type T = Pop<[3, 2, 1]> // T: [3, 2]

Shift<T>:

type Shift<T extends any[]> = T extends [infer _, ...infer O] ? O : never

type T = Shift<[3, 2, 1]> // T: [2, 1]

Reverse<T>

type Reverse<T> = T extends [infer F, ...infer Others]
  ? [...Reverse<Others>, F]
  : []

type T = Reverse<['a', 'b']> // T: ['b', 'a']

使用條件類型來(lái)判斷兩個(gè)類型完全相等

我們也可以使用條件類型來(lái)判斷 A、B 兩個(gè)類型是否完全相等。當(dāng)前社區(qū)上主要有兩種方案:

方案一: 參考 issue。

export type Equal1<T, S> =
  [T] extends [S] ? (
    [S] extends [T] ? true : false
  ) : false

目前該方案的唯一缺點(diǎn)是會(huì)將 any 類型與其它任何類型判為相等。

type T = Equal1<{x:any}, {x:number}> // T: true

方案二: 參考 issue。

export type Equal2<X, Y> =
  (<T>() => T extends X ? 1 : 2) extends
  (<U>() => U extends Y ? 1 : 2) ? true : false

目前該方案的唯一缺點(diǎn)是在對(duì)交叉類型的處理上有一點(diǎn)瑕疵。

type T = Equal2<{x:1} & {y:2}, {x:1, y:2}> // false

以上兩種判斷類型相等的方法見(jiàn)仁見(jiàn)智,筆者在此拋磚引玉。

總結(jié)

到此這篇關(guān)于TypeScript中條件類型精讀與實(shí)踐的文章就介紹到這了,更多相關(guān)TypeScript條件類型內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 小程序?qū)崿F(xiàn)授權(quán)登陸的解決方案

    小程序?qū)崿F(xiàn)授權(quán)登陸的解決方案

    這篇文章主要介紹了小程序?qū)崿F(xiàn)授權(quán)登陸的解決方案,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2018-12-12
  • canvas繪制環(huán)形進(jìn)度條

    canvas繪制環(huán)形進(jìn)度條

    本文主要介紹了canvas繪制環(huán)形進(jìn)度條的示例代碼,具有很好的參考價(jià)值,下面跟著小編一起來(lái)看下吧
    2017-02-02
  • javaScript產(chǎn)生隨機(jī)數(shù)的用法小結(jié)

    javaScript產(chǎn)生隨機(jī)數(shù)的用法小結(jié)

    這篇文章主要介紹了javaScript產(chǎn)生隨機(jī)數(shù)的用法小結(jié),包括JavaScript Math.random()內(nèi)置函數(shù) ,Js 隨機(jī)數(shù)產(chǎn)生6位數(shù)字的代碼,需要的朋友可以參考下
    2018-04-04
  • javascript實(shí)現(xiàn)仿IE頂部的可關(guān)閉警告條

    javascript實(shí)現(xiàn)仿IE頂部的可關(guān)閉警告條

    仿windows IE頂部的敬告工具條,帶關(guān)閉按鈕,設(shè)計(jì)還算精美,你完全可以用到自己的網(wǎng)頁(yè)用于顯示提示等方面,有需要的小伙伴可以參考下。
    2015-05-05
  • 關(guān)于JS中的作用域中的問(wèn)題思考分享

    關(guān)于JS中的作用域中的問(wèn)題思考分享

    這篇文章主要介紹了關(guān)于JS中的作用域中的問(wèn)題思考分享,scope和?closure是?javascript中兩個(gè)非常關(guān)鍵的概念,前者JS用多了還比較好理解而且容易體會(huì)到,而?closure就不一樣了。這玩意是真的很容易迷糊,需要的朋友可以參考下
    2022-04-04
  • js正則表達(dá)式驗(yàn)證郵件地址

    js正則表達(dá)式驗(yàn)證郵件地址

    這篇文章主要介紹了js正則表達(dá)式驗(yàn)證郵件地址,利用javaScript語(yǔ)言實(shí)現(xiàn)一下電子郵件地址驗(yàn)證程序,使用的是JavaScript語(yǔ)言的正則表達(dá)式庫(kù),感興趣的小伙伴們可以參考一下
    2015-11-11
  • javascript中閉包(Closure)詳解

    javascript中閉包(Closure)詳解

    閉包(closure)是Javascript語(yǔ)言的一個(gè)難點(diǎn),也是它的特色,很多高級(jí)應(yīng)用都要依靠閉包實(shí)現(xiàn)。小編之前一直糊里糊涂的,沒(méi)有能夠弄明白JavaScript的閉包到底是什么,有什么用,本文把自己的理解些出來(lái)分享一下,希望不理解JavaScript閉包的朋友們看了之后能夠理解閉包!
    2016-01-01
  • 微信小程序使用藍(lán)牙小插件

    微信小程序使用藍(lán)牙小插件

    這篇文章主要為大家詳細(xì)介紹了微信小程序使用藍(lán)牙小插件,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2019-09-09
  • JavaScript使用遞歸和循環(huán)實(shí)現(xiàn)階乘的實(shí)例代碼

    JavaScript使用遞歸和循環(huán)實(shí)現(xiàn)階乘的實(shí)例代碼

    這篇文章主要介紹了JavaScript使用遞歸和循環(huán)實(shí)現(xiàn)階乘的實(shí)例代碼,代碼簡(jiǎn)單易懂,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友參考下吧
    2018-08-08
  • amd、cmd、esmodule、commonjs區(qū)別詳解

    amd、cmd、esmodule、commonjs區(qū)別詳解

    本文主要介紹了amd、cmd、esmodule、commonjs區(qū)別詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-04-04

最新評(píng)論