TypeScript內(nèi)置工具類型快速入門運用
前言
TS為了方便開發(fā)者使用,在內(nèi)部提供了非常多的工具類型,如Partial、Required、ReadOnly等等,本篇文章主要用于記錄了一些常用的內(nèi)置工具類型的使用及源碼實現(xiàn),以便參考。
提示:以下是本篇文章正文內(nèi)容,下面案例可供參考
一、什么是ts內(nèi)置工具類型
TypeScript 附帶了大量類型,可以幫助進(jìn)行一些常見的類型操作,通常稱為 Utility Types。
二、使用示例
1.Partial
將必填參數(shù)變?yōu)榭蛇x參數(shù)
namespace a { // 示例一: interface A { name: string, age: number, sex: string } // 接口A中的參數(shù)通過內(nèi)置工具類型Partial變成了可選參數(shù) type PartialA = Partial<A>; let a: PartialA = { // 此處傳參可以任意個數(shù),但只能傳已定義的參數(shù) name: '張三', age: 10 } // 示例二: interface Company { id: number, name: string } interface Person { id: number, name: string, company: Company } // 實現(xiàn)DeepPartial,將參數(shù)深度轉(zhuǎn)換成可選參數(shù) type DeepPartial<T> = { [U in keyof T]?: T[U] extends object ? DeepPartial<T[U]> : T[U]; } type PartialPerson = DeepPartial<Person>; let p: PartialPerson = { // 此處的參數(shù)已經(jīng)變?yōu)榭蛇x參數(shù),可根據(jù)自己需要傳遞 id: 10, name: '張三', company: { name: '美團' } } }
Partial的實現(xiàn)源碼
namespace b { interface A { name: string, age: number, sex: string } type Partial<T> = { // 原理: // 1.通過keyof拿到T中的所有key // 2.迭代所有的key // 3.統(tǒng)一加上可選參數(shù)處理 // 4.返回迭代值的類型 [P in keyof T]?: T[P]; } // 測試: type PartialA = Partial<A>; let a: PartialA = { name: '張三', age: 10 } }
2.Required
將所有參數(shù)變?yōu)楸靥?/p>
namespace b { interface Person { name: string, age?: number } // Required將可選參數(shù)變?yōu)楸靥? type RequiredPerson = Required<Person>; let person: RequiredPerson = { name: '張三', age: 10 } }
Required 的實現(xiàn)源碼
namespace c { interface Person { name: string, age?: number } type Required<T> = { // 原理: // 1.通過keyof拿到T中所有的key // 2.迭代所有的key // 3.將可選的參數(shù)變?yōu)楸靥畹膮?shù) // 4.返回迭代屬性及類型 // +?或者?: 代表的是可選,-?: 代表的是不可選 [P in keyof T]-?: T[P]; }; // 測試 type RequiredPerson = Required<Person>; let person: RequiredPerson = { name: '李四', age: 18 } }
3.ReadOnly
將所有參數(shù)變?yōu)橹蛔x
namespace d { interface Person { name: string, age?: number } type ReadonlyPerson = Readonly<Person>; let person: ReadonlyPerson = { name: '張三', age: 10 } // 已經(jīng)變?yōu)橹蛔x屬性,因此此處賦值會報錯 // person.name = '李四'; // 無法分配到 "name" ,因為它是只讀屬性。ts(2540) // person.age = 20; // 無法分配到 "age" ,因為它是只讀屬性。ts(2540) }
Readonly 的實現(xiàn)源碼
namespace e { interface Person { name: string, age?: number } type Readonly<T> = { // 原理: // 1.通過keyof拿到T中所有key // 2.迭代拿到的所有key // 3.通過readonly關(guān)鍵字將所有屬性變?yōu)橹蛔x屬性 // 4.返回迭代屬性及類型 readonly [P in keyof T]: T[P] } // 測試 type ReadonlyPerson = Readonly<Person>; let person: ReadonlyPerson = { name: '張三', age: 10 } // person.name = '李四'; // 無法分配到 "name" ,因為它是只讀屬性。ts(2540) // person.age = 20; // 無法分配到 "age" ,因為它是只讀屬性。ts(2540) }
4.Pick
撿取符合條件的屬性
namespace g { interface Person { name: string, age: number, sex: string } // 參數(shù)一是一個對象,參數(shù)二是要篩選的屬性 type PickPerson = Pick<Person, 'name' | 'sex'>; let person: PickPerson = { name: '張三', // 由于通過Pick只撿取了name和sex屬性,因此此時給sex賦值會報錯 // 不能將類型“{ name: string; age: number; sex: string; }”分配給類型“PickPerson”。 // 對象文字可以只指定已知屬性,并且“age”不在類型“PickPerson”中。ts(2322) // age: 10, sex: '男' } }
Pick 的實現(xiàn)源碼
namespace h { interface Person { name: string, age: number, sex: string } // 原理: // 1.T表示傳入的類型定義。此處指Person接口。 // 2.K表示T的子類型,是一個聯(lián)合類型。此處指'name' | 'age' | 'sex'的子類型 // 3.通過keyof拿到T中的所有key // 3.迭代K,拿到所有傳遞進(jìn)來的子類型 // 4.返回迭代屬性及類型 // 簡單理解:就是從一個對象中,提取一部分屬性組成新的對象 type Pick<T, K extends keyof T> = { [P in K]: T[P]; } // 測試 type PickPerson = Pick<Person, 'name' | 'age'> let person: PickPerson = { name: '張三', age: 10 } }
5.Record
記錄類型:將一個類型的所有屬性值都映射到另一個屬性上并創(chuàng)建新的類型
namespace i { // 1.此處的K主要用來修飾obj對象的key,為string或者number // 2.此處的T用來修飾老的類型 // 3.此處的U用來修飾新的類型 function mapObj<K extends string | number, T, U>(obj: Record<K, T>, callable: (x: T) => U) { // 聲明一個變量 let result: Record<K, U> = <Record<K, U>>{}; // 遍歷傳入的對象 for (const key in obj) { // 通過callback函數(shù)處理result的屬性值 result[key] = callable(obj[key]); } // 返回result return result; } let obj = { num1: 1, num2: 2 }; let callable = (x: number): string => x * 2 + ''; let newObj = mapObj<string | number, number, string>(obj, callable); console.log(newObj); // { num1: '2', num2: '4' } }
Record 的實現(xiàn)源碼
namespace j { type Record<K extends keyof any, T> = { // 原理: // 1.通過keyof拿到所有的K,是一個聯(lián)合類型。string | number | symbol // 2.迭代K,拿到所有的屬性 // 3.返回迭代的屬性及類型 // 注意:此處不能寫成 type Record<K, T> = {}; any代表所有key的聯(lián)合類型。 // 否則會報錯:不能將類型“K”分配給類型“string | number | symbol”。ts(2322) [P in K]: T; }; // 測試 function mapObj<K extends string | number, T, U>(obj: Record<K, T>, callable: (x: T) => U) { // 聲明一個變量 let result: Record<K, U> = <Record<K, U>>{}; // 遍歷傳入的對象 for (const key in obj) { // 通過callback函數(shù)處理result的屬性值 result[key] = callable(obj[key]); } // 返回result return result; } let obj = { num1: 1, num2: 2 }; let callable = (x: number): string => x * 2 + ''; let newObj = mapObj<string | number, number, string>(obj, callable); console.log(newObj); // { num1: '2', num2: '4' } }
總結(jié)
本篇文章只記錄了這幾種內(nèi)置工具類型,更多內(nèi)置類型可前往https://www.typescriptlang.org/docs/handbook/utility-types.html查看。下次再見
到此這篇關(guān)于TypeScript內(nèi)置工具類型快速入門運用的文章就介紹到這了,更多相關(guān)TypeScript內(nèi)置工具內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
js下利用userData實現(xiàn)客戶端保存表單數(shù)據(jù)
對于多數(shù)網(wǎng)頁制作的朋友,實現(xiàn)在客戶端保存在網(wǎng)頁表單上的信息,比較多的是采用Cookie技術(shù)來實現(xiàn),這些功能例如:下拉列表框選擇的選項,文本框輸入的數(shù)據(jù)等。2010-06-06JavaScript數(shù)據(jù)類型和變量_動力節(jié)點Java學(xué)院整理
這篇文章主要為大家詳細(xì)介紹了JavaScript數(shù)據(jù)類型和變量的相關(guān)資料,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-06-06全面接觸神奇的Bootstrap導(dǎo)航條實戰(zhàn)篇
導(dǎo)航條(navbar)在Bootstrap中是一個獨立組件,導(dǎo)航條(navbar)和導(dǎo)航(nav)在Bootstrap中是有明顯區(qū)別的,本文全面接觸神奇的Bootstrap導(dǎo)航條,感興趣的小伙伴們可以參考一下2016-08-08jQuery NProgress.js加載進(jìn)度插件的簡單使用方法
NProgress是基于jquery的,且版本要 >1.8 。這篇文章主要介紹了NProgress.js加載進(jìn)度插件的簡單使用方法,需要的朋友可以參考下2018-01-01