Sean's Blog

An image showing avatar

Hi, I'm Sean

這裡記錄我學習網站開發的筆記
歡迎交流 (ゝ∀・)b

LinkedInGitHub

[Note] TypeScript Course for Beginners - Learn TypeScript from Scratch!

這個 TypeScript 學習大綱包含從基礎到進階的概念,涵蓋了 TypeScript 的基本語法、編譯器配置、高級類型和特性、泛型、裝飾器、命名空間和模組,以及 TypeScript 與其他工具的集成。此大綱適用於想要全面學習 TypeScript 的開發者,從基本開始,然後深入理解 TypeScript 的高級功能。

Get Started

這一部分涵蓋了 TypeScript 的入門和基本概念。可能會涉及以下主題:

  • Using Types:介紹基本的 TypeScript 型別概念
  • Type Assignment and Type Inference:解釋型別指定和型別推斷

💡 The key difference: JavaScript uses “dynamic types” (resolved at runtime), TypeScript uses “static types” (set during development)

TypeScript Basics

這一部分主要介紹 TypeScript 的基本語法和結構,包括基本的 JavaScript 型別和 TypeScript 特有的型別:

  • Numbers, Strings and Booleans:基礎的型別

    1let name: string = 'John';
    2let age: number = 25;
    3let isStudent: boolean = true;
    4
    5function greet(person: string): string {
    6  return `Hello, ${person}!`;
    7}
    8
    9console.log(greet(name));
  • Object Types:對象型別

  • Array Types:陣列型別

  • Tuples:元組

    • 元組的長度是固定的,不可隨意增加或刪除元素
    • 元組中元素的型別和順序必須符合定義
    • 在使用元組時,TypeScript 編譯器會檢查型別和順序,以確保符合定義
    1// 定義一個元組,包含兩個數字
    2let point: [number, number] = [1, 2];
    3
    4// 定義一個包含名字和年齡的元組
    5let person: [string, number] = ['Alice', 25];
  • Enums:枚舉。枚舉的優點包括:

    • 可讀性:使用描述性的名稱代替數字或字串,可以使代碼更容易理解
    • 可維護性:當需要改變常量值時,只需修改枚舉的定義,避免在代碼中到處查找和修改
    • 防止錯誤:使用枚舉可以減少使用非預期值的風險
    1enum Direction {
    2  NORTH = 'N',
    3  EAST = 'E',
    4  SOUTH = 'S',
    5  WEST = 'W',
    6}
    7
    8console.log(Direction.NORTH); // 輸出:"N"

    當定義枚舉時,常見的做法是使用駝峰命名法來定義枚舉的名稱,而使用全大寫來定義枚舉的常量值。這樣做可以在代碼中清楚地標識枚舉的值和其他變數,增強可讀性,有助於區分與普通變數的差異。然而,這不是嚴格規定的規則,而是一種慣例。

  • The Any Type:任意型別

  • Union Types:聯合型別

    • Union Types 用 | 符號將多個型別聯合起來,表明參數可以是其中之一
    1function printValue(value: number | string): void {
    2  if (typeof value === 'number') {
    3    console.log(`The number is ${value}`);
    4  } else {
    5    console.log(`The string is '${value}'`);
    6  }
    7}
    8
    9printValue(42); // 輸出:The number is 42
    10printValue('Hello'); // 輸出:The string is 'Hello'
  • Literal Types:字面量型別

    • Literal Types 允許你使用特定的值作為型別,而不僅僅是使用更通用的型別(如 numberstring

    • 定義一組固定的可能值,並限制其他值的使用

      例如,如果你要定義一個代表方向的變數,它只能是 "north"、"south"、"east" 或 "west",可以使用字面量型別來限定可能的值:

    1type Direction = 'north' | 'south' | 'east' | 'west';
    2
    3function move(direction: Direction): void {
    4  console.log(`Moving in the ${direction} direction`);
    5}
    6
    7move('north'); // 輸出:Moving in the north direction
    8// move('up');  // 錯誤:'up' 不是有效的方向
  • Type Aliases:型別別名

    • 別名可以指向基本型別、聯合型別、元組型別、函數型別,甚至更複雜的結構
    1type Name = string; // 給 string 型別取個別名
    2type StringOrNumber = string | number; // 聯合型別
    3type Direction = 'north' | 'south' | 'east' | 'west'; // 字面量型別
    4// 物件型別別名
    5type Person = {
    6  name: string;
    7  age: number;
    8};
  • Function Return Types and Void:函數返回型別和空型別

    • 在函數定義中指定函數應返回的型別。這有助於確保函數的返回值符合預期,並允許 TypeScript 編譯器進行型別檢查 例如,下面是一個返回數字的函數,它明確指定了返回型別為 number

      1function getSquare(num: number): number {
      2  return num * num;
      3}
      4
      5const result = getSquare(5); // 25
    • void 型別用於表示函數沒有返回值。這在定義那些不需要返回特定值的函數時非常有用,如事件處理器、日誌記錄函數等

      1function logMessage(message: string): void {
      2  console.log(message);
      3}
      4
      5logMessage('Hello, TypeScript');
  • Function Types and Callbacks:函數型別與回呼

    • 函數型別(Function Types)用於指定一個函數的參數和返回值的型別

      1// 定義一個接受兩個數字參數並返回數字的函數型別
      2type MathOperation = (a: number, b: number) => number;
      3
      4// 使用函數型別
      5const add: MathOperation = (x, y) => x + y;
      6const subtract: MathOperation = (x, y) => x - y;
      7
      8console.log(add(5, 3)); // 8
      9console.log(subtract(5, 3)); // 2
    • 回呼函數(Callbacks)是函數型別的一個常見應用。它們允許你將函數作為參數傳遞,並在某些事件或操作完成後調用。 例如,定義一個回呼函數型別,用於處理事件:

      1// 定義一個函數型別,接受一個字串參數且沒有返回值
      2type EventCallback = (event: string) => void;
      3
      4// 使用回呼函數
      5function triggerEvent(callback: EventCallback, eventName: string): void {
      6  console.log(`Triggering event: ${eventName}`);
      7  callback(eventName); // 呼叫回呼函數
      8}
      9
      10const handleEvent: EventCallback = (event) => {
      11  console.log(`Event handled: ${event}`);
      12};
      13
      14triggerEvent(handleEvent, 'ClickEvent');

      在這個例子中,EventCallback 是一個回呼函數型別,定義了它接受一個字串參數且沒有返回值。這種回呼機制在事件驅動程式和異步操作中非常常見。

  • The Unknown Type:未知型別

    • unknown 型別表示一個變數可能是任何型別
    • 它是比 any 更嚴格的通用型別,因為在使用 unknown 型別時,需要執行型別檢查或斷言,才能將其轉換為其他型別
    • unknown 的使用場景包括從外部來源接收數據、處理不確定的值、或者在函數中接受多種可能的參數
    1let value: unknown;
    2
    3value = 42; // 可以是數字
    4value = 'Hello'; // 也可以是字串
    5
    6// 不能直接將 unknown 型別賦值給其他具體型別
    7let name: string;
    8// name = value;  // 錯誤,因為 unknown 不能直接賦值
    9
    10// 必須執行型別檢查
    11if (typeof value === 'string') {
    12  name = value; // 這時可以賦值,因為已確定是字串
    13}

    在這個例子中,我們使用 unknown 表示一個不確定型別的變數。要將 unknown 轉換為具體型別,需要執行型別檢查或型別斷言。

  • The Never Type:永不型別

    • never 型別表示一個函數或表達式永遠不會有返回值,通常是因為它會拋出異常、進行無限循環、或呼叫 process.exit() 這類終止程序的操作

    • never 型別常見於以下幾種情況:

      • 函數拋出異常:當函數永遠不會正常返回,因為它總是拋出異常
      • 無限循環:當函數進入無限循環而不會退出
      • 編譯器不應該到達的地方:通常用於確保代碼的完整性
    1// throwError 總是拋出異常
    2function throwError(message: string): never {
    3  throw new Error(message); // 總是拋出異常
    4}
    5
    6// infiniteLoop 進行無限循環
    7function infiniteLoop(): never {
    8  while (true) {
    9    // 這是一個無限循環
    10  }
    11}
    12
    13// assertNever 用於在 switch 語句中處理不可能的情況
    14function assertNever(x: never): never {
    15  throw new Error('Unexpected value: ' + x);
    16}
    17
    18type Shape = 'circle' | 'square' | 'triangle';
    19
    20function getShapeArea(shape: Shape): number {
    21  switch (shape) {
    22    case 'circle':
    23      return Math.PI * 5 * 5; // 圓形面積
    24    case 'square':
    25      return 25; // 正方形面積
    26    case 'triangle':
    27      return 10; // 三角形面積
    28    default:
    29      assertNever(shape); // 處理不可能的情況
    30  }
    31}

Compiler & Configuration Deep Dive

tsconfig.json 是 TypeScript 專案的配置文件,用於指定編譯器選項、項目文件、型別檢查等。

基礎的 tsconfig.json 配置主要涉及編譯器選項、文件包含與排除,以及編譯目標等,以下是一些基礎選項的簡單說明。

Compiler Options

  • target:指定編譯後的 JavaScript 版本,例如 ES5ES6ES2020 等,這個選項決定了生成的 JavaScript 代碼的兼容性
  • module:指定模組系統,如 CommonJSES6AMD 等,這個選項決定了模組的使用方式。
  • strict:開啟嚴格模式,包含一系列嚴格的型別檢查設定,這是建議的配置,能夠增強型別安全性
  • outDir:指定編譯後輸出的目錄,這個選項決定編譯後的文件位置
  • rootDir:指定源代碼的根目錄,這可以確保正確的目錄結構
  • sourceMap:啟用或禁用源地圖,用於在調試時映射到原始 TypeScript 源碼
  • declaration:生成 TypeScript 的聲明文件(.d.ts),用於外部型別定義
  • removeComments:在編譯後移除註釋
  • noImplicitAny:如果變數未指定型別,則不允許默認為 any。這是一個嚴格模式的選項

File Inclusion and Exclusion

  • include:指定要包含在編譯中的文件或目錄,例如:["src//\*"] 表示包含 src** 目錄及其子目錄的所有文件
  • exclude:指定不包含在編譯中的文件或目錄,例如:["node_modules", "dist"]
1{
2  "compilerOptions": {
3    "target": "ES6",
4    "module": "CommonJS",
5    "strict": true,
6    "outDir": "dist",
7    "rootDir": "src",
8    "sourceMap": true,
9    "declaration": true,
10    "removeComments": true,
11    "noImplicitAny": true
12  },
13  "include": ["src//*"],
14  "exclude": ["node_modules", "dist"]
15}