【JavaScript】Promise(一) —— 理解和使用(是什么、怎么使用、与 Ajax 配合使用、涉及的API)

简介: 【JavaScript】Promise(一) —— 理解和使用(是什么、怎么使用、与 Ajax 配合使用、涉及的API)

一、初识 Promise

1. Promise 是什么?

抽象表达:Promise 是 JS 中进行异步编程的新方案

具体表达:

(1)从语法上来说:Promise 是一个内置构造函数

(2)从功能上来说:Promise 的实例对象可以用来封装一个异步操作,并可以获取其成功 / 失败的值

2. Promise 的理解
  1. Promise 不是回调,它是一个内置的构造函数,是程序员自己new 调用的
  2. new Promise 的时候,要传入一个回调函数,它是同步的回调,会立即在主线程上执行,它被称为executor(执行器)函数
    const p = new Promise(()=>{console.log('小宏');})
    console.log('大白');

executor执行器函数,按顺序执行

b7e8bcb9435b4a45833b2c99b01976f1.png

  1. 每一个Promise实例对象都有3种状态,分别为:初始化(pending)、成功(fulfilled)、失败(rejected)
  2. 每一个Promise实例在刚被new 出来的那一刻,状态都是初始化(pending)

初始化状态:

  const p = new Promise(()=>{})
    console.log(p);

4e25e57884c24c81b48104ea0966b6c5.png

executor 函数会接收到2个参数,它们都是函数,分别用形参:resolve、reject 接收

(1)调用 resolve,会让 Promise 实例状态变为:成功 (filfilled),同时可以指定成功的 value

(2)调用 reject,会让 Promise 实例状态变为:失败 (rejected),同时可以指定失败的 reason

成功的回调:

  const p = new Promise((resolve, reject)=>{resolve('OK')})
    console.log('@', p);

7e621f25cb6347a69449398b9c727d34.png

失败的回调:

  const p = new Promise((resolve, reject)=>{reject('Fail')})
    console.log('@', p);

9867ab4522524c01b697ac1175536824.png

二、Promise 的基本使用

1. 重要语法
  1. new Promise(executor) 构造方法
  2. Promise.prototype.then 方法

2. 基本编码流程

创建 Promise 的实例对象 (pending 状态),传入 executor 函数

在 executor 中启动异步任务 ( 定时器 、ajax请求 )

根据异步任务的结果,做不同处理:

(1)如果异步任务成功了:调用 resolve(value),让 Promise 实例对象状态变为成功 (fulfilled),同时指定成功的 value

(2)如果异步任务失败了:调用 reject(reason),让 Promise 实例对象状态变为失败 (rejected),同时指定失败的 reason

通过 then 方法为 Promise 指定成功、失败的回调函数,来获取成功的 value、失败的 reason

注意:then 方法所指定的:成功的回调、失败的回调,都是异步的回调

成功的回调(异步的):

  const p = new Promise((resolve, reject) => {
        setTimeout(() => {
            resolve('我是成功的数据')
        }, 1000)
    })
    p.then(
        (value) => {console.log('成功了——', value);},  //成功的回调-异步
        (reason) => {console.log('失败了——', reason);}  //失败的回调-异步
    )
    console.log('@主线程');

f54daa3229b44013bff395b347fa844d.png

失败的回调(异步的):

  const p = new Promise((resolve, reject) => {
        setTimeout(() => {
            reject('我是失败错误的信息')
        }, 1000)
    })
    p.then(
        (value) => {console.log('成功了——', value);},  //成功的回调-异步
        (reason) => {console.log('失败了——', reason);}  //失败的回调-异步
    )
    console.log('@主线程');

2bf774b68bf54159a69f69713c9e4051.png

3. 关于状态的注意点

三个状态:

(1)pending:未确定的 ——— 初始状态

(2)fulfilled:成功的 ——— 调用 resolve() 后的状态

(3)rejected:失败的 ——— 调用 reject() 后的状态

两种状态改变(状态只能改变一次)

(1)pending => fulfilled

(2)pending => rejected

状态只能改变一次,不会失败成功都执行:

  const p = new Promise((resolve, reject) => {
        setTimeout(() => {
            reject('我是失败错误的信息')
            resolve('我是成功的数据')
        }, 1000)
    })
    p.then(
        (value) => {console.log('成功了——', value);},  //成功的回调-异步
        (reason) => {console.log('失败了——', reason);}  //失败的回调-异步
    )
    console.log('@主线程');

bb44be6999dd4089bb9c7dd3c49d85d1.png

一个 Promise 指定多个成功/失败回调函数,都会调用

  const p = new Promise((resolve, reject) => {
        setTimeout(() => {
            resolve('我是成功的数据')
            reject('我是失败错误的信息')
        }, 1000)
    })
    p.then(
        (value) => {console.log('成功了1——', value);},  //成功的回调-异步
        (reason) => {console.log('失败了1——', reason);}  //失败的回调-异步
    )
    p.then(
        (value) => {console.log('成功了2——', value);},  //成功的回调-异步
        (reason) => {console.log('失败了2——', reason);}  //失败的回调-异步
    )
    console.log('@主线程');

cd31b18938fc4e139f0875d5f2372d90.png

三、Promise 与 Ajax 配合使用

1. Promise 与 Ajax 结合
  1. 使用原生的 Ajax 充当异步请求,结合 Promise 使用

注: https://api.uixsj.cn/hitokoto/get?type=hitokoto&code=json 是开源 API 文档,打开展示如下

9a12718742c3434097329d5a9b771673.png

  const p = new Promise((resolve, reject) => {
    // 原生 Ajax
        const xhr = new XMLHttpRequest()
        xhr.onreadystatechange = () => {
            if (xhr.readyState === 4) {
                if (xhr.status === 200) {
                    resolve(xhr.response)
                }else{
                    reject('请求出错')
                }
            }
        }
        xhr.open('GET', 'https://api.uixsj.cn/hitokoto/get?type=hitokoto&code=json')
        xhr.responseType = 'json'
        xhr.send()
    })
    p.then(
        (value) => {console.log('成功了——', value);},  //成功的回调-异步
        (reason) => {console.log('失败了——', reason);}  //失败的回调-异步
    )

请求成功:

064b7a086613402cbedd1f424000e02b.png

改错请求地址,请求失败:

a675b085863b4b409b331559cd1253b3.png

2. 封装 Ajax 请求

要求:定义一个sendAjax函数,对xhr的get请求进行封装。


该函数接收两个参数:url (请求地址)、data (参数对象)

该函数返回一个 Promise 实例

(1)若 ajax 请求成功,则 Promise 实例成功,成功的 value 是返回的数据。

(2)若 ajax 请求失败,则 Promise 实例失败,失败的 reason 是错误提示。

  function sendAjax(url, data) {
        return new Promise((resolve, reject) => {
            // 实例xhr
            const xhr = new XMLHttpRequest()
            // 绑定监听
            xhr.onreadystatechange = () => {
                if (xhr.readyState === 4) {
                    if (xhr.status >= 200 && xhr.status < 300) {
                        resolve(xhr.response);
                    } else {
                        reject('请求出了点问题');
                    }
                }
            }
            // 整理参数
            let str = ''
            for (let key in data) {
                str += `${key}=${data[key]}&`
            }
            str = str.slice(0, -1)
            xhr.open('GET', url + '?' + str)
            xhr.responseType = 'json'
            xhr.send()
        })
    }
    const x = sendAjax('https://api.apiopen.top/getJoke', {page:1, count:2, type:'video'})
    x.then(
        (data)=>{console.log('成功了', data);},
        (reason)=>{console.log('失败了', reason);}
    )

请求成功:

f0873509529f479b8f90f18e8bc2808d.png

请求失败:

6f08b604e8da4403b1d927802e6bddcc.png

四、Promise 的 API

1. Promise 构造函数
  new Promise(executor) {}

executor 函数:是同步执行的,(resolve, reject) => {}

resolve 函数:调用 resolve 将 Promise 实例内部状态改为成功 (fulfilled)

reject 函数:调用 reject 将 Promise 实例内部状态改为失败 (rejected)

说明:executor 函数会在 Promise 内部立即同步调用,异步代码放在 executor 函数中

2. Promise.prototype.then 方法
  Promise实例.then(onFulfilled, onRejected)
  1. onFulfilled:成功的回调函数 (value) => {}
  2. onRejected:失败的回调函数 (reason) => {}
  3. 特别注意(难点):then 方法会返回一个新的 Promise 实例对象
3. Promise.prototype.catch 方法
  Promise实例.catch(onRejected)

以下两种是等价的:

  const p = new Promise((resolve, reject)=>{
        setTimeout(()=>{
            reject(-100)
        })
    })
    p.then(
        undefined,
        reason => {console.log('失败了', reason);}
    )
    p.catch(
        reason => {console.log('失败了', reason);}
    )

4085198b46184a379309be974e8e6be2.png

4. Promise.resolve 方法
  Promise.resolve(value)
  1. 说明:用于快速返回一个状态为 fulfilled 或 rejected 的 Promise 实例对象
  2. 备注:value的值可能是:(1)非 Promise 值 (2)Promise 值

传入非Promise值:

  const p = Promise.resolve(100)
  p.then(
        value => {console.log('成功了', value);},
        reason => {console.log('失败了', reason);}
    )

8853cca42ed44bc886df70ddaf070c97.png

传入Promise值:

  const p0 = Promise.reject(-100)
    const p = Promise.resolve(p0)
    p.then(
        value => {console.log('成功了', value);},
        reason => {console.log('失败了', reason);}
    )

15b100e228b645c59e3027d6ecb9cfaf.png

5. Promise.reject 方法
  Promise.reject(reason)
  1. 说明:用于快速返回一个状态为 rejected 的 Promise 实例对象

传入非Promise值:

  const p = Promise.reject(-100)
    p.then(
        value => {console.log('成功了', value);},
        reason => {console.log('失败了', reason);}
    )

531d2d799e984ca3be05be0e14d088af.png

传入Promise值:

    const p0 = Promise.resolve(100)
    const p = Promise.reject(p0)
    p.then(
        value => {console.log('成功了', value);},
        reason => {console.log('失败了', reason);}
    )

c89ffcd8a9b24a72adf50852c7bd36ba.png

6. Promise.all 方法
  Promise.all(promiseArr)
  1. PromiseArr:包含 n 个 Promise 实例的数组
  2. 说明:返回一个新的 Promise 实例,只有所有的 promise 都成功才成功,只要有一个失败就直接是失败

p2失败了,不会再看p3(0.5s后输出结果):

    const p1 = Promise.resolve('a')
    const p2 = new Promise((resolve, reject)=>{
        setTimeout(()=>{
            reject('b')
        }, 500)
    })
    const p3 = new Promise((resolve, reject)=>{
        setTimeout(()=>{
            reject('c')
        }, 2000)
    })
    const x = Promise.all([p1, p2, p3])
    x.then(
        value => {console.log('成功了', value);},
        reason => {console.log('失败了', reason);},
    )

bff752d6fa504cacbc6a1968d42cbec9.png

7. Promise.race 方法
  Promise.race(promiseArr)
  1. promiseArr:包含 n 个 Promise 实例的数组
  2. 说明:返回一个新的 Promise 实例,成功还是失败?要以最先输出结果的 promise 为准

最先输出是成功,所以为成功:

    const p1 = Promise.resolve('a')
    const p2 = new Promise((resolve, reject)=>{
        setTimeout(()=>{
            reject('b')
        }, 500)
    })
    const p3 = new Promise((resolve, reject)=>{
        setTimeout(()=>{
            reject('c')
        }, 2000)
    })
    const x = Promise.race([p1,p2,p3])
    x.then(
        value => {console.log('成功了', value);},
        reason => {console.log('失败了', reason);}
    )

05c285e701264b1da70ec448db2b736e.png

不积跬步无以至千里 不积小流无以成江海

相关文章
|
11月前
|
前端开发 安全 JavaScript
Promise 这个新 API 真香!
本文首发于公众号【沉浸式趣谈】,探讨 ES2025 新特性 `Promise.try()` 的强大功能。该方法可将同步或异步函数统一包装为 Promise,解决同步错误捕获难题,简化代码逻辑。通过实际案例分析,展示其在 API 调用、混合任务链及文件操作中的应用优势。`Promise.try()` 让 Promise 使用更健壮、一致,是开发者工具箱的有力补充。更多技术分享,请访问我的博客 https://yaolifeng.com 或点赞、评论、转发支持!
220 13
|
11月前
|
前端开发 JavaScript NoSQL
使用 Node.js、Express 和 React 构建强大的 API
本文详细介绍如何使用 Node.js、Express 和 React 构建强大且动态的 API。从开发环境搭建到集成 React 前端,再到利用 APIPost 高效测试 API,适合各水平开发者。内容涵盖 Node.js 运行时、Express 框架与 React 库的基础知识及协同工作方式,还涉及数据库连接和前后端数据交互。通过实际代码示例,助你快速上手并优化应用性能。
|
JavaScript 前端开发 API
Vue.js 3:探索组合式API带来的新变革
Vue.js 3:探索组合式API带来的新变革
424 84
|
12月前
|
JavaScript 前端开发 API
JavaScript中通过array.map()实现数据转换、创建派生数组、异步数据流处理、复杂API请求、DOM操作、搜索和过滤等,array.map()的使用详解(附实际应用代码)
array.map()可以用来数据转换、创建派生数组、应用函数、链式调用、异步数据流处理、复杂API请求梳理、提供DOM操作、用来搜索和过滤等,比for好用太多了,主要是写法简单,并且非常直观,并且能提升代码的可读性,也就提升了Long Term代码的可维护性。 只有锻炼思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~
|
JSON 缓存 JavaScript
深入浅出:使用Node.js构建RESTful API
在这个数字时代,API已成为软件开发的基石之一。本文旨在引导初学者通过Node.js和Express框架快速搭建一个功能完备的RESTful API。我们将从零开始,逐步深入,不仅涉及代码编写,还包括设计原则、最佳实践及调试技巧。无论你是初探后端开发,还是希望扩展你的技术栈,这篇文章都将是你的理想指南。
|
JavaScript 前端开发 安全
盘点原生JS中目前最没用的几个功能API
在JavaScript的发展历程中,许多功能与API曾风光无限,但随着技术进步和语言演化,部分功能逐渐被淘汰或被更高效的替代方案取代。例如,`with`语句使代码作用域复杂、可读性差;`void`操作符功能冗余且影响可读性;`eval`函数存在严重安全风险和性能问题;`unescape`和`escape`函数已被`decodeURIComponent`和`encodeURIComponent`取代;`arguments`对象则被ES6的剩余参数语法替代。这些变化体现了JavaScript不断优化的趋势,开发者应紧跟技术步伐,学习新技能,适应新技术环境。
269 10
|
JSON JavaScript 前端开发
深入浅出Node.js:从零开始构建RESTful API
在数字化时代的浪潮中,后端开发作为连接用户与数据的桥梁,扮演着至关重要的角色。本文将引导您步入Node.js的奇妙世界,通过实践操作,掌握如何使用这一强大的JavaScript运行时环境构建高效、可扩展的RESTful API。我们将一同探索Express框架的使用,学习如何设计API端点,处理数据请求,并实现身份验证机制,最终部署我们的成果到云服务器上。无论您是初学者还是有一定基础的开发者,这篇文章都将为您打开一扇通往后端开发深层知识的大门。
352 12
|
JavaScript 前端开发 API
Vue.js 3:深入探索组合式API的实践与应用
Vue.js 3:深入探索组合式API的实践与应用
|
JavaScript NoSQL API
深入浅出Node.js:从零开始构建RESTful API
在数字化时代的浪潮中,后端开发如同一座灯塔,指引着数据的海洋。本文将带你航行在Node.js的海域,探索如何从一张白纸到完成一个功能完备的RESTful API。我们将一起学习如何搭建开发环境、设计API结构、处理数据请求与响应,以及实现数据库交互。准备好了吗?启航吧!
|
JSON JavaScript 前端开发
使用JavaScript和Node.js构建简单的RESTful API
使用JavaScript和Node.js构建简单的RESTful API