溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

TypeScript類型怎么實(shí)現(xiàn)加減乘除

發(fā)布時(shí)間:2023-04-15 14:18:56 來(lái)源:億速云 閱讀:96 作者:iii 欄目:開發(fā)技術(shù)

這篇文章主要講解了“TypeScript類型怎么實(shí)現(xiàn)加減乘除”,文中的講解內(nèi)容簡(jiǎn)單清晰,易于學(xué)習(xí)與理解,下面請(qǐng)大家跟著小編的思路慢慢深入,一起來(lái)研究和學(xué)習(xí)“TypeScript類型怎么實(shí)現(xiàn)加減乘除”吧!

在網(wǎng)上看到這道題目:請(qǐng)用TS類型實(shí)現(xiàn)整除?

type A = Divide<1, 0> // never
type B = Divide<4, 2> // 2
type C = Divide<10, 3> // 3

看完題目,我真的毫無(wú)思路,TS類型還能實(shí)現(xiàn)除法???一臉懵逼的我認(rèn)真地研究了一位叫做  JoeYan大佬的解答:

type Tuple<T extends number, U extends any[] = []> =
    U['length'] extends T ? U : Tuple<T, [...U, any]>

type Subtract<
    A extends number,
    B extends number
> = Tuple<A> extends [...Tuple<B>, ...infer R] ? R['length'] : never

type SmallerThan<
    A extends number,
    B extends number,
    S extends any[] = []
> = S['length'] extends B
    ? false
    : S['length'] extends A
    ? true
    : SmallerThan<A, B, [...S,A]>

type Divide<A extends number, B extends number, S extends any[] = []> =
    B extends 0 ? never : SmallerThan<A, B> extends true ? S['length'] : Divide<Subtract<A, B>, B, [...S, any]>;


type res = Divide<200, 10> // 20

分析

乍一看,真的驚呆了,但是一步一步分析,還是能夠看懂的,本文將整個(gè)研究的過(guò)程記錄了下來(lái):

TS類型沒(méi)有直接提供數(shù)字的加減乘除,所以這位大佬的減法和整除都是通過(guò)數(shù)組長(zhǎng)度計(jì)數(shù)來(lái)實(shí)現(xiàn)的。我平時(shí)體操練習(xí)很少,在沒(méi)看他的解答前,我永遠(yuǎn)不會(huì)想到還能這么玩兒。

Divide

如果要實(shí)現(xiàn)98%10,假設(shè)A是98,B是10,讓A一直減B,直到A小于B,無(wú)法繼續(xù)再減,就能得到整除的結(jié)果。

A能減去9次B,每次進(jìn)行減10的時(shí)候,往S(用來(lái)計(jì)數(shù)的數(shù)組,初始值為空數(shù)組)里面push一個(gè)元素。A減去9次10后,S數(shù)組的長(zhǎng)度是9。此時(shí)A是8,B是10,A小于B,返回S的長(zhǎng)度9。

type Divide<A extends number, B extends number, S extends number[] = []> =
    B extends 0 ? never : SmallerThan<A, B> extends true ? S['length'] : Divide<Subtract<A, B>, B, [...S, any]>;

上面這段代碼的字面意思是:

  • B是否為0,直接返回never

  • A如果小于B,返回S的長(zhǎng)度

  • A如果大于B,我們執(zhí)行A-B,然后我們給S數(shù)組push一個(gè)元素,再次計(jì)算Divide

接下來(lái),讓我們開始逐個(gè)分析。

SmallerThan

SmallerThan用于判斷A是否小于B

type res = SmallerThan<10,2>  // res為false
type res = SmallerThan<2,20> // res為true
type SmallerThan<
    A extends number,
    B extends number,
    S extends any[] = []
> = S['length'] extends B
    ? false
    : S['length'] extends A
    ? true
    : SmallerThan<A, B, [...S,any]>

字面上看起來(lái)是:

  • S的長(zhǎng)度等于B,返回false

  • S的長(zhǎng)度不等于B且S的長(zhǎng)度等于A,返回true

  • S的長(zhǎng)度不等于A和B,將any推入S數(shù)組

接下來(lái)舉例來(lái)看:

type res = SmallerThan<3,2>  

// 首先S['length']=0 ,所以不等于A和B,此時(shí)將any推入數(shù)組,S數(shù)組變成[any]
// 接下來(lái)S['length'] =1,還是B等于A和B,此時(shí)繼續(xù)將A放入數(shù)組,S數(shù)組變成[any,any]
// 此時(shí)S['length'] = 2,所以得出S的長(zhǎng)度等于B,返回false

總之,S的長(zhǎng)度是一次一次的累加的,先等于誰(shuí)的長(zhǎng)度,誰(shuí)就更小。 如果S的長(zhǎng)度先等于B的長(zhǎng)度,那么就是A>B。如果S的長(zhǎng)度先等于A的長(zhǎng)度,就是A<B。

Tuple

作用是將數(shù)字轉(zhuǎn)成數(shù)組,且數(shù)組的長(zhǎng)度等于數(shù)字的大小

type Tuple<T extends number, U extends any[] = []> =
    U['length'] extends T ? U : Tuple<T, [...U, any]>
type res4 = Tuple<3> // [any, any, any]‘

// 基本上和上面的SmallerThan差不多,就是不夠長(zhǎng)度,就push一個(gè)any進(jìn)去

Subtract

顧名思義,獲取A-B的值

type Subtract<
    A extends number,
    B extends number
> = Tuple<A> extends [...Tuple<B>, ...infer R] ? R : never

type res3 = Subtract<10,8> // [any, any]

type Subtract<
    A extends number,
    B extends number
> = Tuple<A> extends [...Tuple<B>, ...infer R] ? R['length'] : never


type res3 = Subtract<20,10> // 10

// 一開始把A轉(zhuǎn)換長(zhǎng)度為20的數(shù)組,B轉(zhuǎn)換成長(zhǎng)度為10的數(shù)組,然后讓ts自己去infer,A的長(zhǎng)度等于B的長(zhǎng)度加上多少長(zhǎng)度的數(shù)組,然后返回R的長(zhǎng)度

最后

前面已經(jīng)實(shí)現(xiàn)了整除和減法,本著練習(xí)的態(tài)度,讓我們?cè)賹?shí)現(xiàn)一下乘法和加法。

加法

仿照前面的Subtract,不難實(shí)現(xiàn):

type Add<A extends number, B extends number > = [...Tuple<A>,...Tuple<B>] extends [... infer T] ? T['length']:never

乘法

接下來(lái),讓我們實(shí)現(xiàn)一下乘法:

5*6 可以看作,5+5+5+5+5+5。

A*B,也就是A要累加自己B次。如果我們每進(jìn)行一次加法,就讓被乘數(shù)B減一,直到被乘數(shù)B為0,也就完成了累加。

type Tuple<T extends number, U extends any[] = []> =
    U['length'] extends T ? U : Tuple<T, [...U, any]>
    
type Mutiply<A extends number, B extends number, S extends any[] = []> = B extends 0 ? S['length'] : Mutiply<A, Subtract<B, 1>, [...S, ...Tuple<A>]>

type res = Mutiply<5,6> //30

坑點(diǎn)

type Tuple<T extends number, S extends any[] = []> = S['length'] extends T ? S : Tuple<T, [...S, any]> // 不報(bào)錯(cuò)
// type Tuple<T extends number, S extends any[] = []> = T extends S['length'] ? S : Tuple<T, [...S, any]> 
// 不能寫成T extends S['length'],ts會(huì)報(bào)遞歸可能是無(wú)限的

感謝各位的閱讀,以上就是“TypeScript類型怎么實(shí)現(xiàn)加減乘除”的內(nèi)容了,經(jīng)過(guò)本文的學(xué)習(xí)后,相信大家對(duì)TypeScript類型怎么實(shí)現(xiàn)加減乘除這一問(wèn)題有了更深刻的體會(huì),具體使用情況還需要大家實(shí)踐驗(yàn)證。這里是億速云,小編將為大家推送更多相關(guān)知識(shí)點(diǎn)的文章,歡迎關(guān)注!

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI