搜尋
首頁web前端js教程分享一個不錯的TS面試題(含3層),看看能答到第幾層!

分享一個不錯的TS面試題(含3層),看看能答到第幾層!

最近遇見一個不錯的 TS 面試題,分享一下。

這題有 3 個層次,我們一層層來看。

第一層的要求是這樣的:

實作一個zip 函數,對兩個陣列的元素依序兩兩合併,例如輸入[1,2,3] , [4,5,6] 時,返回[[1,4], [2,5],[3,6]]

這層就是每次各從兩個陣列取一個元素,合併之後放到陣列裡,然後繼續處理下一個,遞歸進行這個流程,直到陣列為空即可。

function zip(target, source) {
  if (!target.length || !source.length) return [];

  const [one, ...rest1] = target;
  const [other, ...rest2] = source;

  return [[one, other], ...zip(rest1, rest2)];
}

結果是對的:

第一層還是比較簡單的,然後我們來看第二層要求:

#給這個zip 函數定義ts 類型(兩種寫法)

函數的定義有兩種形式:

#直接透過function 宣告函數:

function func() {}

和宣告匿名函數接著賦值給變數:

const func = () => {}

而參數和傳回值的型別都是數組,只是具體型別不知道,可以寫unknown[]。

所以兩個函數類型的定義就是這樣的:

#也是直接function 宣告函數類型和interface 聲明函數類型接著加到變數類型上兩種。

因為具體元素類型不知道,所以用 unknown。

這裡可能會問any 和unknown 的差別:

any 和unknown 都可以接收任何型別:

但是any 也可以賦值給任何類型,但unknown 不行。

這裡只是用來接收其他類型, 所以 unknown 比any 更合適一些,更安全。

這一層也是比較基礎的ts 語法,第三層就上了難度了:

用型別程式實作精確的型別提示,例如參數傳入[1, 2,3], [4,5,6],那回傳值的型別要提示出[[1,4], [2,5],[3,6]]

這裡要求回傳值類型是精確的,我們就要根據參數的類型來動態產生回傳值類型。

也就是這樣:

宣告兩個型別參數 Target、Source,約束為 unknown[],也就是元素型別任意的陣列型別。

這兩個類型參數分別是傳入的兩個參數的型別。

傳回值透過 Zip 計算得出。

接著要實現 Zip 的高階類型:

傳入的型別參數分別是兩個陣列類型,我們同樣要從中提取每個元素合併在一起。

提取元素可以用模式匹配的方式:

所以這個類型就可以這樣定義:

type Zip<One extends unknown[], Other extends unknown[]> =
    One extends [infer OneFirst,...infer Rest1]
      ? Other extends [infer OtherFirst, ...infer Rest2]
        ? [[OneFirst, OtherFirst], ...Zip<Rest1, Rest2>]
        : []
      : [];

分別擷取兩個陣列的第一個元素,建構成新陣列。然後對剩下的數組遞歸進行這樣的處理,直到數組為空。

這樣就實現了我們想要的高階類型:

但你把它加到函數上會錯誤是回傳值:

因為宣告函數的時候都不知道參數是啥,自然計算不出Zip 的值,所以這裡會類型不符:

#那該怎麼辦?

可以用函數重載解決:

ts 支援函數重載,可以寫多個同名函數的類型的類型定義,最後寫函數的實現,這樣用到這個函數的時候會根據參數的型別來匹配函數型別。

我們用了類型程式設計的那個函數透過這種方式寫就不會報錯了。

我們使用下來看看:

#

咋回傳值的型別不對呢?

其實這時候匹配的函數型別是對的,只不過推導出的不是字面量類型。

這時候可以加個 as const。

但加上as const 會推導出readonly [1,2,3]

這樣型別就不符合了,所以要在型別參數的宣告上也加上readonly:

但這樣Zip 函數的型別又不匹配了。

難道要把所有用到這個類型的地方都加上 readonly 麼?

不用,我們 readonly 的修飾去掉不就行了?

Typescript 有內建的高階類型readonly:

#可以把索引類型的每個索引都加上readonly 修飾:

但沒有提供去掉readonly 修飾的高階類型,我們可以自己實作:

用對映類型的語法建構個新索引類型,加上個-readonly 就是去掉readonly 修飾的意思。

有的同學可能問了,陣列型別也是索引型別?

是,索引類型是聚合多個元素的類型,所以物件、陣列、class 都是。

所以我們把它用在數組上自然也是可以的:

#(準確來說叫元組,元組是元素個數固定的陣列)

那我們只要在傳入Zip 之前,用Mutable 去掉readonly 就可以了:

再來試試看:

##大功告成!現在傳回值的類型就對了。

但還有個問題,如果不是直接傳入字面量,是推導不出字面量類型的,這時候貌似就不對了:

可我們不都宣告重載類型了麼?

如果推導不出字面量類型,應該要匹配這個呀:

但實際上它匹配的還是第一個:

#這時候其實只要調換下兩個函數型別的順序就可以了:

這時字面量參數的情況依然也是對的:

為什麼呢? ######因為###重載函數的型別是從上到下依序匹配,只要匹配到一個就應用。 #########非字面量的情況,型別是 number[],能符合 unknown[] 的那個型別,所以那個函數型別生效了。 ###############而字面量的情況,推導出的是readonly [1,2,3],帶有readonly 所以不匹配unknown[],繼續往下匹配,就匹配到了帶有類型參數的那個函數類型。 ######這樣兩種情況就都應用了合適的函數型別。 ######全部程式碼是這樣的:###
type Zip<One extends unknown[], Other extends unknown[]> = One extends [
  infer OneFirst,
  ...infer Rest1
]
  ? Other extends [infer OtherFirst, ...infer Rest2]
    ? [[OneFirst, OtherFirst], ...Zip<Rest1, Rest2>]
    : []
  : [];

type Mutable<Obj> = {
  -readonly [Key in keyof Obj]: Obj[Key];
};

function zip(target: unknown[], source: unknown[]): unknown[];

function zip<Target extends readonly unknown[], Source extends readonly unknown[]>(
  target: Target,
  source: Source
): Zip<Mutable<Target>, Mutable<Source>>;

function zip(target: unknown[], source: unknown[]) {
  if (!target.length || !source.length) return [];

  const [one, ...rest1] = target;
  const [other, ...rest2] = source;

  return [[one, other], ...zip(rest1, rest2)];
}

const result = zip([1, 2, 3] as const, [4, 5, 6] as const);

const arr1 = [1, 2, 3];
const arr2 = [4, &#39;5&#39;, 6];

const result2 = zip(arr1, arr2);

ts playground 地址

总结

今天我们做了一道综合的 ts 面试题,一共有三层:

第一层实现 js 的逻辑,用递归或者循环都能实现。

第二层给函数加上类型,用 function 声明类型和 interface 声明函数类型两种方式,参数和返回值都是 unknown[]。

第三层是用类型编程实现精准的类型提示,这一层需要拿到参数的类型,通过提取元素的类型并构造出新的数组类型返回。还要通过函数重载的方式来声明类型,并且要注意重载类型的声明顺序。

as const 能够让字面量推导出字面量类型,但会带有 readonly 修饰,可以自己写映射类型来去掉这个修饰。

其实这也是我们学习 ts 的顺序,我们先要能把 js 逻辑写出来,然后知道怎么给函数、class 等加 ts 类型,之后学习类型编程,知道怎么动态生成类型。

其中类型编程是 ts 最难的部分,也是最强大的部分。攻克了这一层,ts 就可以说学的差不多了。

【相关推荐:javascript学习教程

以上是分享一個不錯的TS面試題(含3層),看看能答到第幾層!的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述
本文轉載於:掘金社区。如有侵權,請聯絡admin@php.cn刪除
5个常见的JavaScript内存错误5个常见的JavaScript内存错误Aug 25, 2022 am 10:27 AM

JavaScript 不提供任何内存管理操作。相反,内存由 JavaScript VM 通过内存回收过程管理,该过程称为垃圾收集。

实战:vscode中开发一个支持vue文件跳转到定义的插件实战:vscode中开发一个支持vue文件跳转到定义的插件Nov 16, 2022 pm 08:43 PM

vscode自身是支持vue文件组件跳转到定义的,但是支持的力度是非常弱的。我们在vue-cli的配置的下,可以写很多灵活的用法,这样可以提升我们的生产效率。但是正是这些灵活的写法,导致了vscode自身提供的功能无法支持跳转到文件定义。为了兼容这些灵活的写法,提高工作效率,所以写了一个vscode支持vue文件跳转到定义的插件。

巧用CSS实现各种奇形怪状按钮(附代码)巧用CSS实现各种奇形怪状按钮(附代码)Jul 19, 2022 am 11:28 AM

本篇文章带大家看看怎么使用 CSS 轻松实现高频出现的各类奇形怪状按钮,希望对大家有所帮助!

Node.js 19正式发布,聊聊它的 6 大特性!Node.js 19正式发布,聊聊它的 6 大特性!Nov 16, 2022 pm 08:34 PM

Node 19已正式发布,下面本篇文章就来带大家详解了解一下Node.js 19的 6 大特性,希望对大家有所帮助!

浅析Vue3动态组件怎么进行异常处理浅析Vue3动态组件怎么进行异常处理Dec 02, 2022 pm 09:11 PM

Vue3动态组件怎么进行异常处理?下面本篇文章带大家聊聊Vue3 动态组件异常处理的方法,希望对大家有所帮助!

聊聊如何选择一个最好的Node.js Docker镜像?聊聊如何选择一个最好的Node.js Docker镜像?Dec 13, 2022 pm 08:00 PM

选择一个Node​的Docker镜像看起来像是一件小事,但是镜像的大小和潜在漏洞可能会对你的CI/CD流程和安全造成重大的影响。那我们如何选择一个最好Node.js Docker镜像呢?

聊聊Node.js中的 GC (垃圾回收)机制聊聊Node.js中的 GC (垃圾回收)机制Nov 29, 2022 pm 08:44 PM

Node.js 是如何做 GC (垃圾回收)的?下面本篇文章就来带大家了解一下。

【6大类】实用的前端处理文件的工具库,快来收藏吧!【6大类】实用的前端处理文件的工具库,快来收藏吧!Jul 15, 2022 pm 02:58 PM

本篇文章给大家整理和分享几个前端文件处理相关的实用工具库,共分成6大类一一介绍给大家,希望对大家有所帮助。

See all articles

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

AI Hentai Generator

AI Hentai Generator

免費產生 AI 無盡。

熱門文章

R.E.P.O.能量晶體解釋及其做什麼(黃色晶體)
3 週前By尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.最佳圖形設置
3 週前By尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.如果您聽不到任何人,如何修復音頻
3 週前By尊渡假赌尊渡假赌尊渡假赌

熱工具

mPDF

mPDF

mPDF是一個PHP庫,可以從UTF-8編碼的HTML產生PDF檔案。原作者Ian Back編寫mPDF以從他的網站上「即時」輸出PDF文件,並處理不同的語言。與原始腳本如HTML2FPDF相比,它的速度較慢,並且在使用Unicode字體時產生的檔案較大,但支援CSS樣式等,並進行了大量增強。支援幾乎所有語言,包括RTL(阿拉伯語和希伯來語)和CJK(中日韓)。支援嵌套的區塊級元素(如P、DIV),

SecLists

SecLists

SecLists是最終安全測試人員的伙伴。它是一個包含各種類型清單的集合,這些清單在安全評估過程中經常使用,而且都在一個地方。 SecLists透過方便地提供安全測試人員可能需要的所有列表,幫助提高安全測試的效率和生產力。清單類型包括使用者名稱、密碼、URL、模糊測試有效載荷、敏感資料模式、Web shell等等。測試人員只需將此儲存庫拉到新的測試機上,他就可以存取所需的每種類型的清單。

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Safe Exam Browser

Safe Exam Browser

Safe Exam Browser是一個安全的瀏覽器環境,安全地進行線上考試。該軟體將任何電腦變成一個安全的工作站。它控制對任何實用工具的訪問,並防止學生使用未經授權的資源。

PhpStorm Mac 版本

PhpStorm Mac 版本

最新(2018.2.1 )專業的PHP整合開發工具