實(shí)用工具類(lèi)型

以下實(shí)用工具類(lèi)型適用于2.X及以上的typescript版本(為方便直觀的看出結(jié)果近上,下面配置為嚴(yán)格模式贵试,當(dāng)出現(xiàn)問(wèn)題時(shí)編譯不通過(guò)歌粥,直接報(bào)錯(cuò))

Partial<T>

構(gòu)造類(lèi)型T沮脖,并將它所有的屬性設(shè)置為可選的癞季。它的返回類(lèi)型表示輸入類(lèi)型的所有子類(lèi)型劫瞳,T一般為對(duì)象(ps:對(duì)象才有用到此方法的意義,其他簡(jiǎn)單類(lèi)型沒(méi)有使用他沒(méi)有什么意義)

type Partial<T> = { [P in keyof T]?: T[P] | undefined; }

Partial<T>.png
import React, { Component } from 'react';

interface Interface {
    a: number;
    b: number;
}

export default class PartialInfo extends Component {

    render() {

// 變量類(lèi)型為對(duì)象
        // 正確顯示值
        const partial1: Partial<Interface> = {};
        const partial2: Partial<Interface> = {
            a: 1
        };
        const partial3: Partial<Interface> = {
            a: 1,
            b: 1
        };

        // 不屬于Interface的key值
        const partial4: Partial<Interface> = {
            // 不能將類(lèi)型“{ c: number; }”分配給類(lèi)型“Partial<Interface>”绷柒。
            //   對(duì)象文字可以只指定已知屬性志于,并且“c”不在類(lèi)型“Partial<Interface>”中。
            c: 1  // ?
        };


// 變量類(lèi)型為string(無(wú)具體意義废睦,僅證明可以輸入)
        // Partial<T>中的T可以是任意類(lèi)型不必拘泥于對(duì)象
        const partial5: Partial<string> = '';

        console.log(partial1, partial2, partial3, partial4, partial5);

        return (
            <div>Partial</div>
        );
    }
}

使用場(chǎng)景

  • interfaceA與 interfaceB 在代碼中同時(shí)會(huì)被【使用】伺绽,并且interfaceB需要屬性, interfaceA中全部存在(interfaceA與 interfaceB 在代碼中同時(shí)會(huì)被【使用】嗜湃,并且interfaceB需要屬性奈应, interfaceA中全部存在)
  • eg:對(duì)表格進(jìn)行篩選,篩選條件的類(lèi)型可為Partial<T>
  • eg:
    // 賬號(hào)屬性
    interface AccountInfo { name: string email: string age: number vip: 0|1 // 1 是vip 购披,0 是非vip }
    // 當(dāng)我們需要渲染一個(gè)賬號(hào)表格時(shí)杖挣,我們需要定義
    const accountList: AccountInfo[] = []
    // 但當(dāng)我們需要查詢(xún)過(guò)濾賬號(hào)信息,需要通過(guò)表單刚陡,
    // 但明顯我們可能并不一定需要用到所有屬性進(jìn)行搜索惩妇,此時(shí)可以定義
    const model: Partial<AccountInfo> = { name: '', vip: undefind }

Readonly<T>

構(gòu)造類(lèi)型T,并將它所有的屬性設(shè)置為readonly筐乳,也就是說(shuō)構(gòu)造出的類(lèi)型的屬性不能被再次賦值歌殃。

import React, { Component } from 'react';

interface Todo {
    title: string;
}

export default class ReadonlyInfo extends Component {

// 對(duì)象

    todo: Readonly<Todo> = {
        title: 'one',
    };
    this.todo.title = 'Hello'; 
    render() {
// 對(duì)象
        console.log('todo', this.todo);
        return (
            <div>Readonly</div>
        );
    }
}

Readonly.png
string1: Readonly<string> = '234';
this.string1 = '34';
T為string類(lèi)型.png

Object.freeze

方法可以?xún)鼋Y(jié)一個(gè)對(duì)象。一個(gè)被凍結(jié)的對(duì)象再也不能被修改蝙云;凍結(jié)了一個(gè)對(duì)象則不能向這個(gè)對(duì)象添加新的屬性氓皱,不能刪除已有屬性,不能修改該對(duì)象已有屬性的可枚舉性勃刨、可配置性波材、可寫(xiě)性,以及不能修改已有屬性的值身隐⊥⑶【此外,凍結(jié)一個(gè)對(duì)象后該對(duì)象的原型也不能被修改抡医《阋颍】??【不寫(xiě)進(jìn)ppt】??freeze()返回和傳入的參數(shù)相同的對(duì)象早敬。

import React, { Component } from 'react';
interface Todo {
    title: string;
}
export default class FreezeInfo extends Component {

// 對(duì)象
    todo = {
        title: 'one'
    };
  
    freezeObj = (todo: Todo) => {
        Object.freeze(todo);
        todo.value = '1';
        console.log('todo', todo)  
    }

    render() {
// 對(duì)象
        this.freezeObj(this.todo)
        return (
            <div>freeze</div>
        );
    }
}

Object.freeze.png

Object.freeze僅對(duì)對(duì)象起作用忌傻,對(duì)其他類(lèi)型不起作用(包括數(shù)組)

freezeString = (todo: string) => {
        Object.freeze(todo);
        todo = '1';
        console.log('todo', todo)  
}
this.freezeObj(this.todo)
Object.freeze為字符串.png

Record<K,T>

構(gòu)造一個(gè)類(lèi)型,其屬性名的類(lèi)型為K搞监,屬性值的類(lèi)型為T(mén)水孩。將某個(gè)類(lèi)型的屬性映射到另一個(gè)類(lèi)型上,當(dāng)K為某些固定值而非范圍性類(lèi)型時(shí),需要全部映射,定義的屬性名類(lèi)型必須全部都有琐驴,屬性值看是否可選

K的類(lèi)型為number | string | symbol

type Record<K extends string | number | symbol, T> = { [P in K]: T; }

import React, { Component } from 'react';

interface PageInfo {
    a: number;
    b?: number;
}
type Page = 'key1' | 'key2' | 'key3';
type PageIn = number | string;

export default class RecordInfo extends Component {

    render() {

        // 屬性名為具體值
        const page1: Record<Page, PageInfo> = {
            key1: { a: 1, b: 2 },
            key2: { a: 1 },
            key3: { a: 1 },
        };

        // 屬性名為類(lèi)型范圍值
        // 此處屬性名類(lèi)型K中使用的是固定的變量俘种,也可以使用入number秤标、string等的范圍性類(lèi)型,那么此刻屬性名隨意宙刘、個(gè)數(shù)隨意
        const page2: Record<PageIn, PageInfo> = {
            1: { a: 1, b: 2 },
            2: { a: 1, b: 2 },
            key1: { a: 1 },
            key2: { a: 1 },
        };

        console.log(page1, page2)
        return (
            <div>Partial</div>
        );
    }
}

Pick<T,K>

從類(lèi)型T中挑選部分屬性K來(lái)構(gòu)造類(lèi)型苍姜。T中必須包含K中的所有屬性,K中的屬性悬包,必須在對(duì)象中全部擁有(ps衙猪,T中可以只有K中的屬性)

import React, { Component } from 'react';

interface Todo {
    title: string;
    description: string;
    completed: boolean;
}
type TodoPreview = Pick<Todo, 'title' | 'completed'>;

export default class PickInfo extends Component {

// 對(duì)象
    todo: TodoPreview = {
        title: 'Clean room',
        completed: false,
    };

    render() {
        console.log('todo', this.todo)
        return (
            <div>Pick</div>
        );
    }
}
T中未包含K.png
對(duì)象未全部擁有K.png

Exclude<T,U>

從類(lèi)型T中剔除所有可以賦值給U的屬性,然后構(gòu)造一個(gè)類(lèi)型布近。允許T中沒(méi)有U屬性的情況存在

Exclude<T,U>.png

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

import React, { Component } from 'react';

interface Obj1 {
    a: string;
    b: number;
}
interface Obj2 {
    a: string;
    b: number;
    c: string;
}
interface Obj3 {
    b: string;
    c: number;
}
interface Obj4 {
    a: string;
    b: string;
    c: string;
}

type T0 = Exclude<"a" | "b" | "c", "a">;  // "b" | "c"
type T1 = Exclude<"a" | "b" | "c", "a" | "b">;  // "c"
type T2 = Exclude<string | number | (() => void), Function>;  // string | number
type T3 = Exclude<Obj1 | Obj2 | Obj3, Obj1>;  // Obj3
type T4 = Exclude<Obj1 | Obj4 | Obj3, Obj1>;  // Obj4 | Obj3

export default class ExcludeInfo extends Component {
    
    render() {
        return (
            <div>Exclude</div>
        );
    }
}

  • () => void與Function為何換了位置垫释,結(jié)果就完全不同了,參見(jiàn)上篇【類(lèi)型】中的函數(shù)賦值
  • type T3 = Extract<Obj1 | Obj2 | Obj3, Obj1>; // 值為Obj1 | Obj2原因撑瞧,參見(jiàn)上篇【類(lèi)型】中的對(duì)象賦值

使用場(chǎng)景

interface的信息與其他interface有關(guān)棵譬,會(huì)隨著其他的interface變化而變化

Extract<T,U>

從類(lèi)型T中提取所有可以賦值給U的類(lèi)型,然后構(gòu)造一個(gè)類(lèi)型预伺。簡(jiǎn)而言之就是T订咸、U中有相同的部分,就提取出來(lái)扭屁,作為類(lèi)型值

Extract<T,U>.png

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

import React, { Component } from 'react';
interface Obj1 {
    a: string;
    b: number;
}
interface Obj2 {
    a: string;
    b: number;
    c: string;
}
interface Obj3 {
    b: string;
    c: number;
}

interface Obj4 {
    a: string;
    b: string;
    c: string;
}

type T0 = Extract<"a" | "b" | "c", "a" | "f">;  // "a"
type T1 = Extract<string | number | (() => void), Function>;  // () => void
type T2 = Extract<string | number | Function, (() => void)>;  // never
type T3 = Extract<Obj1 | Obj2 | Obj3, Obj1>;  // Obj1 | Obj2
type T4 = Extract<Obj1 | Obj4 | Obj3, Obj1>; // Obj1

export default class ExtractInfo extends Component {

    render() {
        return (
            <div>Extract</div>
        );
    }
}
  • () => void與Function為何換了位置算谈,結(jié)果就完全不同了,參見(jiàn)上篇【類(lèi)型】中的函數(shù)賦值
  • type T3 = Extract<Obj1 | Obj2 | Obj3, Obj1>; // 值為Obj1 | Obj2原因料滥,參見(jiàn)上篇【類(lèi)型】中的對(duì)象賦值

使用場(chǎng)景

interface的信息與其他interface有關(guān)然眼,會(huì)隨著其他的interface變化而變化

NonNullable<T>

從類(lèi)型T中剔除null和undefined,然后構(gòu)造一個(gè)類(lèi)型葵腹。

type T0 = NonNullable<string | number | undefined>;  // string | number
type T1 = NonNullable<string[] | null | undefined>;  // string[]

ReturnType<T>

由函數(shù)類(lèi)型T的返回值類(lèi)型構(gòu)造一個(gè)類(lèi)型高每。

import React, { Component } from 'react';

const f1 = () => {
    return {
        a: 1,
        b: 'string'
    }
}

type T0 = ReturnType<() => string>;  // string
type T1 = ReturnType<(s: string) => void>;  // void
type T2 = ReturnType<(<T>() => T)>;  // unknown
type T3 = ReturnType<(<T extends U, U extends number[]>() => T)>;  // number[]
type T4 = ReturnType<typeof f1>;  // { a: number, b: string }

export default class ReturnTypeInfo extends Component {
    b:T7 = 6;
    render() {
        let a:T2 = 3;
        a = '3';
        console.log('a', a)
        console.log('b', this.b)
        return (
            <div>ReturnType</div>
        );
    }
}

當(dāng)返回的值為unknown時(shí),就代表他是任意值践宴,你可以給此類(lèi)型的參數(shù)賦給任意值鲸匿,不會(huì)報(bào)錯(cuò),但并不代表他不進(jìn)行編譯了阻肩,它相當(dāng)于一個(gè){},任何對(duì)象都屬于這個(gè){}带欢,所以賦為任何值都不會(huì)報(bào)錯(cuò)。

有兩個(gè)特殊的值烤惊,一個(gè)為any乔煞,另一個(gè)為never,分別為柒室,編譯時(shí)略過(guò)檢查渡贾,和不存在一個(gè)值滿(mǎn)足當(dāng)前條件,這兩個(gè)值作為參數(shù)傳入,默認(rèn)是不報(bào)錯(cuò)的

type T5 = ReturnType<any>;  // any
type T6 = ReturnType<never>;  // never

ReturnType接收的參數(shù)T必須為一個(gè)函數(shù)類(lèi)型雄右,傳入其他類(lèi)型值時(shí)報(bào)錯(cuò)空骚,當(dāng)T為Function時(shí)報(bào)錯(cuò)纺讲。

type T7 = ReturnType<string>;  // Error
type T8 = ReturnType<Function>;  // Error
傳入值為Function.png

用一種簡(jiǎn)單的思維去思考為什么Function不能傳入,回歸到定義本身【由函數(shù)類(lèi)型T的返回值類(lèi)型構(gòu)造一個(gè)類(lèi)型囤屹“旧酰】,如果Function沒(méi)有返回值肋坚,要怎么辦呢则涯?是不是這么一問(wèn)就直接能解決上文為何Function作為傳入值報(bào)錯(cuò)

Required<T>

構(gòu)造一個(gè)類(lèi)型,使類(lèi)型T的所有屬性為required冲簿。即便在interface中寫(xiě)的是可選類(lèi)型粟判,但如果使用了Required,那么所有的屬性都變成必須的峦剔,沒(méi)有例外

import React, { Component } from 'react';
interface Obj {
    a?: number;
    b?: string;
};

export default class RequiredInfo extends Component {

    render() {
        const obj1: Obj = { a: 5 }; // OK
        const obj2: Required<Obj> = { a: 5 }; // Error: property 'b' missing
        console.log('obj1', obj1);
        console.log('obj2', obj2);
        return (
            <div>Required</div>
        );
    }
}

Required.png
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末档礁,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子吝沫,更是在濱河造成了極大的恐慌呻澜,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,525評(píng)論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件惨险,死亡現(xiàn)場(chǎng)離奇詭異羹幸,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)辫愉,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,203評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門(mén)栅受,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人恭朗,你說(shuō)我怎么就攤上這事屏镊。” “怎么了痰腮?”我有些...
    開(kāi)封第一講書(shū)人閱讀 164,862評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵而芥,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我膀值,道長(zhǎng)棍丐,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,728評(píng)論 1 294
  • 正文 為了忘掉前任沧踏,我火速辦了婚禮歌逢,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘悦冀。我一直安慰自己趋翻,他們只是感情好睛琳,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,743評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布盒蟆。 她就那樣靜靜地躺著踏烙,像睡著了一般。 火紅的嫁衣襯著肌膚如雪历等。 梳的紋絲不亂的頭發(fā)上讨惩,一...
    開(kāi)封第一講書(shū)人閱讀 51,590評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音寒屯,去河邊找鬼荐捻。 笑死,一個(gè)胖子當(dāng)著我的面吹牛寡夹,可吹牛的內(nèi)容都是我干的处面。 我是一名探鬼主播,決...
    沈念sama閱讀 40,330評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼菩掏,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼魂角!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起智绸,我...
    開(kāi)封第一講書(shū)人閱讀 39,244評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤野揪,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后瞧栗,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體斯稳,經(jīng)...
    沈念sama閱讀 45,693評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,885評(píng)論 3 336
  • 正文 我和宋清朗相戀三年迹恐,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了挣惰。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,001評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡殴边,死狀恐怖通熄,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情找都,我是刑警寧澤唇辨,帶...
    沈念sama閱讀 35,723評(píng)論 5 346
  • 正文 年R本政府宣布,位于F島的核電站能耻,受9級(jí)特大地震影響赏枚,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜晓猛,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,343評(píng)論 3 330
  • 文/蒙蒙 一饿幅、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧戒职,春花似錦栗恩、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,919評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)乳乌。三九已至,卻和暖如春市咆,著一層夾襖步出監(jiān)牢的瞬間汉操,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,042評(píng)論 1 270
  • 我被黑心中介騙來(lái)泰國(guó)打工蒙兰, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留磷瘤,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,191評(píng)論 3 370
  • 正文 我出身青樓搜变,卻偏偏與公主長(zhǎng)得像采缚,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子挠他,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,955評(píng)論 2 355