为什么要使用 Promise?
在探讨 Promise 之前,我们先来看一下为什么需要它的出现。
JavaScript 中有一个重要概念 - 异步 (async),它允许我们在执行耗时任务时,不必等待程序完成,而是继续执行下面的代码,直到任务完成再通知。常用的异步操作有:文件操作、数据库操作、AJAX 以及定时器等。
JavaScript 有两种实现异步的方式:
第一种:回调函数 callback function
在 ES6 promise 出现之前,通常使用回调函数 (callback) 实现异步操作。但使用回调函数 callback 存在一个明显的缺点,当需要执行多个异步操作时,代码会不断往内嵌套,这种情况通常被称为“回调地狱”(callback hell)。
callback(() => {
  console.log("Hello!");
  callback(() => {
    console.log("Hello!");
    callback(() => {
      console.log("Hello!");
      callback(() => {
        console.log("Hello!");
      }, 200);
    }, 200);
  }, 200);
}, 200);
而为了解决这种问题,就出现了第二种方法 - promise。
什么是 Promise?
上一段提到 Promise 出现的原因,这一段我们来看那到底 Promise 是什么。
Promise 照英文上的意思,是约定、承诺,它代表的意涵是这个约定请求会在未来每个时刻返回数据给调用者。在 MDN 文件中,Promise 是用来表示 一个异步操作的最终完成(或失败)及其结果值。
怎么使用 Promise
Promise 是一个 构造函数,我们需要通过 new 关键字创建一个 Promise。而 Promise 会接收一个函数作为参数,这个函数又称为 executor,executor 会立即执行。如下方代码,若丢入浏览器开发者工具执行,console 的结果会立刻被打打打印出来。
new Promise((resolve, reject) => {
  console.log("executor 立即执行"); // executor 立即执行
});
而 executor 函数,会再接受另外两个函数参数
- resolve 实现函数:如下方代码,请求成功的例子,正确的时候会调用 resolve 函数,并返回结果。
- reject 拒绝函数:如下方代码,请求失败的例子,失败的时候会调用 reject 函数,并返回结果。
function requestData(url) {
  return new Promise((resolve, reject) => {
    setTimeout(() => {
      if (url === "explainthis.io") {
        resolve("hello welcome to explainthis");
      } else {
        reject("it is not explainthis");
      }
    }, 3000);
  });
}
// 1. 请求成功
requestData("explainthis.io").then((res) => {
  console.log(res); //hello welcome to explainthis
});
//2. 请求失败
requestData("explainthis.com").catch((e) => console.log(e)); //it is not explainthis
Promise 的状态
一个 Promise 一定会处于以下三种状态的其中一种
- pending:初始状态,执行了 executor,但还在等待中。
- fulfilled:表示操作完成,执行 resolve 函数。
- rejected:表示操作失败,执行 reject 函数。
then 的使用
延续前段谈到的,异步用第一种 callback 做法很容易有 callback hell 的产生,而使用 Promise 的好处则可以避免这种难以阅读的写法。
Promise 可以用一种链式 (chaining) 的方式将这些异步操作串连,如下方代码示例,我们可以通过 then 来将等完成之后的操作串起。
function requestData(url) {
  return new Promise((resolve, reject) => {
    setTimeout(() => {
      if (url === "explainthis.io") {
        resolve("hello welcome to explainthis");
      } else {
        reject("it is not explainthis");
      }
    }, 3000);
  });
}
requestData("explainthis.io")
  .then((res) => {
    console.log(res); //hello welcome to explainthis
    return 1;
  })
  .then((res) => {
    console.log(res); // 1
    return 2; //hello welcome to explainthis
  })
  .then((res) => {
    console.log(res); // 2
  });
- then 方法可以接受两个参数,一个为成功的回调,另一个为失败的回调
function requestData(url) {
  return new Promise((resolve, reject) => {
    setTimeout(() => {
      if (url === "explainthis.io") {
        resolve("hello welcome to explainthis");
      } else {
        reject("it is not explainthis");
      }
    }, 0);
  });
}
requestData("explainthis.com").then(
  (res) => {
    console.log(res);
  },
  (reason) => {
    console.log(reason);
  }
);
//it is not explainthis
错误处理
Promise 的一个好处是错误处理,最简单的方式是在加上一个 catch 来捕捉错误,并执行一些错误处理代码。如下方代码,如果请求失败,例如由于网络故障,则 Promise 会被拒绝。在这种情况下,catch 方法将捕获错误,并输出错误讯息。
fetch("https://explainthis.com/data")
  .then((response) => response.json())
  .then((data) => {
    console.log(data);
  })
  .catch((error) => {
    console.error("oops!", error);
  })
  .finally(() => {
    console.log("close loader");
  });
finally 方法
如果有加上 finally,那 Promise 状态不论是 fulfilled 还是 rejected 都会需要执行 finally 方法。finally 是 Promise 处理流程中一个非常有用的方法,它可以帮助我们在不管 Promise 是否成功的状态下,执行一定必要的操作。
使用场景例如,一进入页面要先从服务器 fetch 数据,等待的同时会显示 loading 的画面,而最后不论是否有拿到数据,我们都需要把 loader 关闭。这时候,关闭 loader 的逻辑,就很适合放在 finally 中。如下方代码:
fetch("https://explainthis.com/data")
  .then((response) => response.json())
  .then((data) => {
    console.log(data);
  })
  .catch((error) => {
    console.error(error);
  })
  .finally(() => {
    console.log("close loader");
  });
什么是 async/await?
async/await 是一种基于 Promise 之上的语法糖,比 Promise 的写法更像是同步操作。
首先,我们会使用 async 关键字将函数标记为异步函数,异步函数就是指返回值为 Promise 对象的函数。
在异步函数中我们可以调用其他的异步函数,不过不是使用 then(),而是使用 await 语法,await 会等待 Promise 完成之后直接返回最终的结果。
async function getData() {
  const res = await fetch("https://getsomedata");
  const data = await res.json();
  console.log(data);
}
getData();
该文章在 2024/8/11 3:08:56 编辑过