[JavaScript理論學習] 什麼是Promise (含如何判斷一個值是Promise)

来源:https://www.cnblogs.com/Yiero/archive/2023/09/08/17686480.html
-Advertisement-
Play Games

# 什麼是Promise (含如何判斷一個值是Promise) > 本文旨在對 Promise 的規範進行解釋, 便於讀者在學習 Promise 的過程中梳理 Promise 之間的操作關係, 不對具體的代碼實現和Promise用法進行解釋. > > 比如, 為什麼 [[MDN-await]](ht ...


什麼是Promise (含如何判斷一個值是Promise)

本文旨在對 Promise 的規範進行解釋, 便於讀者在學習 Promise 的過程中梳理 Promise 之間的操作關係, 不對具體的代碼實現和Promise用法進行解釋.

比如, 為什麼 [MDN-await] 中要提及一個 thenable 對象, 而且這個 thenable 對象還可以和 Promise 實例一樣使用 await 等待處理, 這就涉及到了下麵的內容.

由於筆者編程水平的限制, 不可避免存在錯漏或者語意不清的地方.

Promise A+ 規範

參考資料: [Promises/A+]

在 ES6 之前,社區已經有了 Promise A+ 規範, 該規範定義了 Promise 的行為和介面. 根據規範, 任何具有 .then() 方法的函數或對象都可以被認為是一個 Promise ,並且可以進行 Promise 之間的操作。

這個具有 .then() 方法的函數/對象被稱為 thenable 對象, 你可以在 [MDN-Promise#thenable][Promises/A+] 查閱到相關資料.

如果讀者您熟悉 ES6 中的 Promise , 那麼對 Promise A+ 規範一定不陌生, 因為 ES6 中的 Promise 就是基於 Promise A+ 規範的官方實現和拓展. 我們可以將 ES6 的 Promise 稱為 Promise 對象, 在 ES6 之前, 第三方庫實現的 Promise A+ 規範對象稱為 thenable 對象.

該規範旨在於解決回調地獄和非同步實現不統一的問題. 早在 ES6 之前, 就有很多第三方庫遵守和支持這個規範. 比如 jQuery 中的 $.ajax() / $.get() 等方法返回的就是一個 JQuery 實現的 thenable 對象.

ES6 Promise

基於 Promise A+ 規範, 在 ES6 中新增了一個構造函數 Promise, 通過實例化 Promise (new Promise()) 可以新建一個 Promise 對象, 這個對象是一個符合 Promise A+ 規範的對象 .

如果為了便於理解, 我們可以將 Promise 對象 簡單理解為一個繼承了 thenable 對象 的對象.

不過在 Promise A+ 的基礎上, ES6還拓展了更多的功能, 比如 .catch() 方法, .finally() 方法 , 靜態方法 Promise.all() 等等, 具體可以查閱 MDN-Promise .

需要另外瞭解的一點是, .catch( (error) => {} )方法本質上就是第一個參數傳入了空參數的.then( undefined, (error) => {} )方法.

最小實現的 Promise 和最大實現的 Promise

綜上所述, 我們可以將 Promise A+ 規範規定的 Promise 稱為最小實現的 Promise, 也就是 thenable 對象; 將 ES6 的 Promise 成為最大實現的 Promise, 即 Promise 對象.

如果要檢測一個值是否為最小實現的 Promise , 只需要檢測是否函數/對象, 並且存在 .then() 方法即可.

如果要檢測一個值是否為最大實現的 Promise, 則只需要在上面的檢測的基礎上, 添加一個 .finally() 方法的檢測.

什麼是Promise

在 Promise A+ 規範中, Promise 就是一個具有 .then() 方法的函數或者對象.

在 ES6(ES2015) 中, Promise 是一個構造函數, 通過這個構造函數可以實例化一個符合 Promise A+ 規範的對象.

在 ES7(ES2016) 及其之後的版本, 還可以使用 await / async 去調用所有符合 Promise A+ 規範的對象, 包括一些第三方庫自己實現的符合 Promise A+ 規範的對象.

只要是符合 Promise A+ 規範的 Promise , 那麼它們之間就可以互相操作.

工具函數, 檢測一個對象是否為Promise

通常不會直接使用類似value instanceof Promise的判斷, 而是給予 Promise A+ / ES6 Promise 規範判斷.

最小限定的檢測, 檢測是否為 thenable 對象.

/**
 * 判斷傳入參數是否為 Promise (最小實現)
 * @param { any } value
 * @return { boolean } 是否為 Promise
 * @description 代碼源於第三方庫 is-promise
 * @tutorial https://www.npmjs.com/package/is-promise
 * */
function isPromise( value ) {
	return !!value
		&& (typeof value === 'object' || typeof value === 'function')
		&& typeof value.then === 'function';
}

取消了 typeof value === 'function' 的判斷, 因為通過 new Promise() 實例化的值一定是一個對象.

如果想更嚴格一下, 還可以將 .catch() 方法也加進判斷, 也有些工具函數只判斷 .then() 方法和 .catch() 方法是否存在, 這隻看使用者是想如何限定 Promise 的範圍.

/**
 * 判斷傳入參數是否為 Promise (最大實現)
 * @param { any } value
 * @return { boolean } 是否為 Promise
 * */
function isStrictPromise( value ) {
	return !!value
		&& typeof value === 'object'
		&& typeof value.then === 'function'
		&& typeof value.finally === 'function';
}

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 編輯my.ini或者my.cnf文件 #### 清空binlog信息 ```sql #查看現存的binlog文件列表 show master logs; #重置清空binlog文件 reset master; #重置清空後 重新查看現存的binlog文件列表 是否都被清空了 show master ...
  • 本文分享自華為雲社區《GaussDB AP是如何執行SQL的》,作者:yd_270088468。 前言 介紹GaussDB AP各組件是如何協調工作的,會著重介紹SQL引擎。 1、SQL引擎組件和SQL生命周期 Parser: 詞法/語法分析模塊。詞法分析會從SQL字元串中解析出一個個單詞,作為語法 ...
  • GaussDB的遷移場景越來越多,也越來越複雜,所以我們會不斷地進行探索和創新,讓我們的方案更完善,遷移過程更平滑。 ...
  • SqlServer 單用戶解決方案 USE master; GO DECLARE @SQL VARCHAR(MAX); SET @SQL='' SELECT @SQL=@SQL+'; KILL '+RTRIM(SPID) - FROM master..sysprocesses WHERE dbid= ...
  • 在上一篇文章中,我們介紹了彈性資料庫連接失效的背景,並探討了HikariCP連接池探活策略的相關內容。在本文中,我們將會繼續探討另一個線上常用的連接池——Druid,併為您介紹如何在使用Druid時實現最佳實踐的彈性資料庫連接池探活策略。 ...
  • 原文地址:https://www.mssqltips.com/sqlservertip/3598/troubleshooting-transactional-replication-latency-issues-in-sql-server/ 問題 我安裝了幾個SQL Server 2012實例的集群 ...
  • sidebar: auto # Android 調試橋 (adb) Android 調試橋 (adb) 是一種功能多樣的命令行工具,可讓您與設備進行通信。adb 命令可用於執行各種設備操作,例如安裝和調試應用。adb 提供對 Unix shell(可用來在設備上運行各種命令)的訪問許可權。它是一種客戶 ...
  • 在Service中使用系統dialog彈框,但是無法覆蓋全部,底部菜單依然可以被點擊,在某些場景下是不符合需求的 getDialog().getWindow().setType(WindowManager.LayoutParams.TYPE_SYSTEM_ERROR); 顯然是 dialog 的層級 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...