」工欲善其事,必先利其器。「—孔子《論語.錄靈公》
首頁 > 程式設計 > TypeScript 嚴格類型 - 部分完整靜態類型

TypeScript 嚴格類型 - 部分完整靜態類型

發佈於2024-08-06
瀏覽:630

TypeScript strictly typed - Part full static types

在本系列文章的前一部分中,我们讨论了安全可为空性。

现在我们将解释并解决 TypeScript 默认行为的第三个问题,也是最后一个问题:动态类型的剩余部分

我们将涵盖:

  • 动态类型的残余
  • 实际平等检查
  • 条件中没有隐式转换
  • 条件简写
  • 字符串和数字不能混合

动态类型的残余

TypeScript 应该是一个“静态类型检查器”,与 JavaScript 不同,JavaScript 中的类型是深度动态的。

但在本系列文章的前一部分中,我们还解释了 TypeScript 是作为 JavaScript 的超集构建的。

所以问题是:JavaScript 动态类型系统的某些部分仍然保留在 TypeScript 中。因此,我们将解释如何抑制这些剩余行为以实现完全静态类型

实际平等检查

  • ESLint: eqeqeq
  • 生物群落:可疑.noDoubleEquals(推荐)

这个问题最好的例子是相等性检查。在 JavaScript 中,== 并不完全是相等检查,而是等价检查:

1 == "1"; // true

尽管类型不同,但一些转换规则会起作用,因此 JavaScript 能够比较值。它可能会导致很多错误,因为规则细节很难记住,有时很奇怪,并且在所有动态语言(例如 PHP)中并不完全相同。

这些等价性检查仅在 JavaScript 等动态类型语言中才有意义。从我们决定使用 TypeScript 的那一刻起,只应使用实际的相等检查(类型和值)。

1 === "1"; // false

eqeqeq lint 规则强制执行它。

来自 Java、C# 或 Rust 等语言的人应该特别小心这个问题,因为 JavaScript 或 TypeScript 中的 == 与这些语言中的含义不同。在 JavaScript 和 TypeScript 中,需要第三个 = 才能实现相同的行为。

条件中没有隐式转换

  • ESLint:@typescript-eslint/strict-boolean-expressions
  • 生物群落:缺少规则

认为现在情​​况安全吗?不幸的是不是,因为转换可以是隐式的:

let tax: number | undefined = 0;

if (tax) {
  console.log("Process payment");
}
if (!tax) {
  throw new Error();
}

上面的例子相当于:

let tax: number | undefined = 0;

if (tax == true) {
  console.log("Process payment");
}
if (tax == false) {
  throw new Error();
}

如您所见,存在隐式 ==,因此仍然会发生转换:0 不等于 true,它等于 false。因此,尽管税费是有效值,它也会出错。

严格布尔表达式 lint 规则不允许此类隐式条件,并强制执行实际检查:

let tax: number | undefined = 0;

if (tax !== undefined) {
  console.log("Process payment");
}
if (tax === undefined) {
  throw new Error();
}

对于习惯了 JavaScript 快速条件的人来说,这可能是最繁琐的规则之一,但从长远来看,这只是在 Java、C# 或 Rust 等其他语言中执行操作的正常方法。

如配置部分所示,禁用allowNumber和allowString子选项对于避免所有错误非常重要。

唯一允许的例外是对象和数组:这些情况是安全的,因为与字符串和数字相反,它们没有假值。所以下面的还是可以的:

let movie: Movie | undefined;
if (movie) {}
if (!movie) {}

注意:switch 语句已经安全,因为它们在内部使用 ===。

条件简写

  • ESLint:@typescript-eslint/prefer-nullish-coalescing(在风格类型检查中)
  • 生物群落:缺少规则

严格布尔表达式 lint 规则注意条件检查是类型安全的,但除了 if:
之外,还有其他条件语法

const movieRating = userRating || 5;

// Which is a shorter version of:
const movieRating = userRating == true ? userRating : 5;

如果用户评分为 0,则 0 相当于 false,因此评分将为 5 而不是 0。

现代 JavaScript 可以避免这种情况:

const movieRating = userRating ?? 5;

// Which is a shorter version of:
const movieRating = userRating !== undefined && userRating !== null
  ? userRating
  : 5;

它可以通过prefer-nullish-coalescing lint 规则强制执行。

注意 ??不应在任何地方使用: ||在使用布尔值时仍然相关。

字符串和数字不能混合

  • ESLint:
    • 首选模板
    • @typescript-eslint/restrict-plus-operands(在推荐类型检查中)
    • @typescript-eslint/restrict-template-expressions(在推荐类型检查中)
  • 生物群落:
    • style.useTemplate(推荐)
    • 缺少其他规则

在 JavaScript 中,运算符既可用于数字的数学加法,也可用于字符串连接。它会导致错误。

"There is "   3   1   "Matrix movies"; // 31
"There is "   (3   1)   "Matrix movies"; // 4

运算符应保留用于数学加法。或者至少,它应该仅用于相同类型的数据,这是restrict-plus-operands lint 规则强制执行的。

现代 JavaScript 中的模板字符串应该用于字符串连接,prefer-template lint 规则强制执行:

const movie = `Everything everywhere all at once`;
`${movie} is the best movie!`;

相反,模板字符串中只能使用字符串,这是restrict-template-expressions lint 规则强制执行的。

如果混合类型是实际想要的,则转换应该是显式的:

const total = 3;
`There is ${total.toFixed()} Matrix movies`;

注意模板字符串可以嵌套:

const total = 3;
`There is ${total.toFixed()} Matrix movie${total > 1 ? "s" : ""}`;

结论

这是本系列文章的结尾。您可以关注我的帐户(本页右上角的按钮),了解有关 TypeScript 或 Angular 等其他主题的其他帖子何时发布。

您想联系我?摘要中提供了说明。

版本聲明 本文轉載於:https://dev.to/cyrilletuzi/typescript-strictly-typed-part-4-full-static-types-8bc?1如有侵犯,請聯絡[email protected]刪除
最新教學 更多>
  • 如何使用預先匯入的套件啟動 python 或 ipython。
    如何使用預先匯入的套件啟動 python 或 ipython。
    不必每次執行 python 或 ipython 時都必須編寫“import os”或其他常見包,此腳本將啟動 python 並導入這些包 #!/usr/bin/env bash # -----------------------------------------------------------...
    程式設計 發佈於2024-11-06
  • python中的資料類型
    python中的資料類型
    內建資料型別 文字類型:字串 數值類型:int、float、complex 序列類型:清單、元組、範圍 映射類型:字典 集合類型:集合、凍結集合 布林類型:bool 二進位類型:位元組、位元組數組、記憶體視圖 無型別:無型別 程式 a=「阿維納什」 印刷(類型(a)) b=45 印刷(類型(b)) ...
    程式設計 發佈於2024-11-06
  • JavaScript 中的提升
    JavaScript 中的提升
    什么是 JavaScript 提升? 提升是指 JavaScript 中在执行之前进行声明的过程。首先处理变量和函数声明。因此,即使变量在声明之前被引用,也不会导致错误,而是返回 undefined。对于函数声明,整个函数被提升,这意味着它可以在代码中定义之前使用。此过程在执行开始...
    程式設計 發佈於2024-11-06
  • 如何在 Go 中同時選擇緩衝發送和無緩衝接收通道?
    如何在 Go 中同時選擇緩衝發送和無緩衝接收通道?
    同時選擇 Go 發送和接收通道在 Go 中,可以使用 select 語句在通道上執行非阻塞 I/O 操作。然而,在處理緩衝發送通道(chan
    程式設計 發佈於2024-11-06
  • 如何將列表列表轉換為統一的 NumPy 陣列?
    如何將列表列表轉換為統一的 NumPy 陣列?
    將清單清單轉換為NumPy 陣列資料分析中的一個常見任務是將清單清單轉換為NumPy 陣列高效率的數值運算。這個數組可以透過將每個列表分配給一行來形成,列表中的每個元素佔據一列。 選項 1:陣列陣列如果子清單有不同的長度,適當的方法是建立陣列的陣列。這保留了清單清單的原始結構,從而可以輕鬆檢索特定元...
    程式設計 發佈於2024-11-06
  • 前端的頂級設計模式
    前端的頂級設計模式
    在过去的几个月里,我为前端开发人员分享了一些流行的设计模式。其中包括 Singleton、Facade、Observer、Publisher/Subscriber 等模式。今天,我想总结一下这些模式的一些要点和好处,以及如何使用它们来改进您的前端开发流程。 什么是设计模式 设计模式是...
    程式設計 發佈於2024-11-06
  • ServBay版本.pdate公告
    ServBay版本.pdate公告
    我們很高興地宣布新版本 1.4.4 已經到來!讓我們來看看新增的備受期待的新功能。 新功能 CA和憑證管理: 統一SSL憑證管理平台:全新的憑證管理平台,旨在簡化憑證申請與管理流程。 ServBay User CA 和 ServBay Public CA: 引入 ...
    程式設計 發佈於2024-11-06
  • Spring框架中的控制反轉
    Spring框架中的控制反轉
    控制反转(IoC)和依赖注入(DI)是Spring框架中的两个基本概念。传统上,对象负责创建和管理它们自己的依赖关系。然而,IoC 通过将对象创建和依赖管理的控制权移交给像 Spring 这样的框架来翻转这一责任。 这种转变有几个优点: 更简单的实现交换:只需对代码库进行最小的更改即可交换不同的实现...
    程式設計 發佈於2024-11-06
  • 使用 React 建立遞歸檔案系統:深入探討
    使用 React 建立遞歸檔案系統:深入探討
    簡介:在 React 中建構遞歸檔案系統 在現代 Web 開發中,建立互動式動態檔案系統是常見的要求。無論是管理文件、組織專案或建構複雜的資料結構,擁有強大的文件系統都至關重要。在這篇文章中,我們將探討如何在 React 中建立遞歸檔案系統,並專注於可以新增、重新命名或刪除的嵌套資...
    程式設計 發佈於2024-11-06
  • SQL 查詢速度慢?使用此技術提高應用程式的效能
    SQL 查詢速度慢?使用此技術提高應用程式的效能
    挑戰 在我的應用程式(React Spring Boot Oracle)中,處理大型資料集導致處理時間極為緩慢。我需要一種解決方案來提高效能而不影響準確性或完整性。 解決方案:NTILE 並行處理 NTILE 是一個功能強大的 SQL 視窗函數,旨在將結果集劃分為...
    程式設計 發佈於2024-11-06
  • 關於測試覆蓋率的真相
    關於測試覆蓋率的真相
    一個強而有力的真理。 看下面這段簡單明了的程式碼: function sum(a, b) { return a b; } 現在,讓我們為它寫一些測試: test('sum', () => { expect(sum(1, 2)).toBe(3); expect(...
    程式設計 發佈於2024-11-06
  • 為什麼我的 OpenGL 三角形無法在 Go 中渲染?調查頂點緩衝區問題。
    為什麼我的 OpenGL 三角形無法在 Go 中渲染?調查頂點緩衝區問題。
    Go 中的OpenGL 頂點緩衝區問題在Go 中嘗試使用OpenGL 顯示三角形時,使用者遇到了頂點緩衝區問題緩衝區無法渲染形狀。 Go 程式碼源自於教程,但與 C 程式碼不同的是,它沒有產生任何輸出。 問題原因問題的根本原因位於傳遞給 vertexAttrib.AttribPointer() 的參...
    程式設計 發佈於2024-11-06
  • 為什麼在 Linux 32 位元發行版上的 Go 程式中設定 `ulimit -n` 會導致「參數無效」錯誤?
    為什麼在 Linux 32 位元發行版上的 Go 程式中設定 `ulimit -n` 會導致「參數無效」錯誤?
    如何在 Go 程式中設定 ulimit -n? 問題使用者嘗試在 Go 程式中設定 ulimit -n使用 setrlimit 和 getrlimit 系統呼叫將其限制在程式內而不是全域。然而,在嘗試設定該值時出現錯誤,提示「參數無效」。 解決方案發現問題是由於 Linux 32 的 Getrlim...
    程式設計 發佈於2024-11-06
  • 如何在Python中創建無限深度的動態嵌套字典?
    如何在Python中創建無限深度的動態嵌套字典?
    未定義深度的動態嵌套字典在涉及複雜多層資料結構的場景中,經常會遇到變數嵌套字典的需求水平。雖然硬編碼插入語句是一種潛在的解決方案,但當事先未知嵌套深度時,這種方法是不切實際的。 要克服此限制,請考慮利用 Python 的 collections.defaultdict,它允許動態建立字典。可以使用下...
    程式設計 發佈於2024-11-06
  • Python 變得強大:輕鬆程式設計的初學者指南
    Python 變得強大:輕鬆程式設計的初學者指南
    Python 是一門強大的程式語言,文法簡單,應用廣泛。安裝 Python 後,可以學習其基本語法,包括變數賦值、資料類型和流程控制。實戰案例中,我們透過蒙特卡羅模擬計算圓周率,展示了 Python 在數值計算中的能力。此外,Python 擁有豐富的函式庫,涵蓋機器學習、資料分析和網路開發等領域,體...
    程式設計 發佈於2024-11-06

免責聲明: 提供的所有資源部分來自互聯網,如果有侵犯您的版權或其他權益,請說明詳細緣由並提供版權或權益證明然後發到郵箱:[email protected] 我們會在第一時間內為您處理。

Copyright© 2022 湘ICP备2022001581号-3