博客
关于我
ES6躬行记(23)——Promise的静态方法和应用
阅读量:452 次
发布时间:2019-03-06

本文共 3685 字,大约阅读时间需要 12 分钟。

一、静态方法

Promise有四个静态方法,分别是resolve()、reject()、all()和race(),本节将着重分析这几个方法的功能和特点。

1)Promise.resolve()

此方法有一个可选的参数,参数的类型会影响它的返回值,具体可分为三种情况,其中有两种情况会创建一个新的已处理的Promise实例,还有一种情况会返回这个参数。

(1)当参数为空或非thenable时,返回一个新的状态为fulfilled的Promise。

(2)当参数为thenable时,返回一个新的Promise,而它的状态由自身的then()方法控制。

(3)当参数为Promise时,将不做修改,直接返回这个Promise。

下面用一个例子演示这三种情况:

let tha = {  then(resolve, reject) {    resolve("thenable");  }};//参数为空Promise.resolve().then(function(value) {  console.log(value);    //undefined});//参数为非thenablePromise.resolve("string").then(function(value) {  console.log(value);    //"string"});//参数为thenablePromise.resolve(tha).then(function(value) {  console.log(value);    //"thenable"});//参数为PromisePromise.resolve(new Promise(function(resolve) {  resolve("Promise");})).then(function(value) {  console.log(value);    //"Promise"});

2)Promise.reject()

此方法能接收一个参数,表示拒绝理由,它的返回值是一个新的已拒绝的Promise实例。与Promise.resolve()不同,Promise.reject()中所有类型的参数都会原封不动的传递给后续的已拒绝的回调函数。

Promise.reject("rejected").catch(function (reason) {  console.log(reason);          //"rejected"});var p = Promise.resolve();Promise.reject(p).catch(function (reason) {  reason === p;                 //true});

第一次调用Promise.reject()的参数是一个字符串,第二次的参数是一个Promise,catch()方法中的回调函数接收到的正是这两个参数。

3)Promise.all()

此方法和接下来要讲解的Promise.race()都可用来监控多个Promise,当它们的状态发生变化时,这两个方法会给出不同的处理方式。

Promise.all()能接收一个可迭代对象,其中可迭代对象中的成员必须是Promise,如果是字符串、thenable等非Promise的值,那么会自动调用Promise.resolve()转换成Promise。Promise.all()的返回值是一个新的Promise实例。

(1)当可迭代对象中的所有成员都是已完成的Promise时,新的Promise的状态为fulfilled。而各个成员的决议结果会组成一个数组,传递给后续的已完成的回调函数。

var p1 = Promise.resolve(200),  p2 = "fulfilled";Promise.all([p1, p2]).then(function (value) {  console.log(value);          //[200, "fulfilled"]});

(2)当可迭代对象中的成员有一个是已拒绝的Promise时,新的Promise的状态为rejected。并且只会处理到这个已拒绝的成员,接下来的成员都会被忽略,其决议结果会传递给后续的已拒绝的回调函数。

var p1 = Promise.reject("error"),  p2 = "fulfilled";Promise.all([p1, p2]).catch(function (reason) {  console.log(reason);         //"error"});

4)Promise.race()

此方法和Promise.all()有很多相似的地方,如下所列。

(1)能接收一个可迭代对象。

(2)成员必须是Promise,对于非Promise的值要用Promise.resolve()做转换。

(3)返回值是一个新的Promise实例。

新的Promise实例的状态也与方法的参数有关,当参数的成员为空时,其状态为pending;当参数不为空时,其状态是最先被处理的成员的状态,并且此成员的决议结果会传递给后续相应的回调函数。

var p1 = new Promise(function(resolve) {  setTimeout(() => {    resolve("fulfilled");  }, 200);});var p2 = new Promise(function(resolve, reject) {  setTimeout(() => {    reject("rejected");  }, 100);});Promise.race([p1, p2]).catch(function (reason) {  console.log(reason);      //"rejected"});

在p1和p2的执行器中都有一个定时器。由于后者的定时器会先执行,因此通过调用Promise.race([p1, p2])得到的Promise实例,其状态和p2的相同,而p2的决议结果会作为拒绝理由被catch()方法中的回调函数接收。

根据前面的分析可以得出,Promise.all()能处理一个或多个受监控的Promise,而Promise.race()只能处理其中的一个。

二、应用

1)Promise和生成器

用Promise和生成器实现一个运行器,可以取代冗长的Promise链,以一种更直观的方式控制异步,类似于下面这样。

function load() {  return new Promise(function(resolve, reject) {    resolve("success");  });}run(function* ()    {  var result = yield load();  console.log(result);        //"success"});

在load()函数内部,执行的是异步操作,由于处在run运行器中,因此它的决议结果可通过赋值语句直接给到result。这种工作模式完全颠覆了过去用回调函数接收异步操作结果的模式。为此,ES8规范特地引入了两个关键字:async和await,支持这种便捷的工作模式,下面改写上一个示例,用新语法实现相同的功能。

(async function() {  var result = await load();  console.log(result);       //"success"})();

关于run运行器的工作原理和ES8的新语法,限于篇幅原因,此处不再展开说明。

2)与传统异步操作的组合

以往需要用回调函数接收异步处理结果的操作,现在都能改成Promise的模式。以图像加载为例,当图像载入成功时,会触发load事件;而当失败时,会触发error事件。如果后续又有异步操作,那么就只能在这两个事件中处理,但这么一来,就会形成难以控制的回调金字塔。而改用Promise后,就能链式的处理后续的操作,如下所示。

function preImg(src) {  return new Promise(function (resolve, reject) {    var img = new Image();    img.src = src;    img.onload = function(){      resolve(this);    };    img.onerror = function(){      reject(this);    };  });};preImg("img/page.png").then(function(value) {  console.log(value);});

转载地址:http://huafz.baihongyu.com/

你可能感兴趣的文章
Node-RED中使用range范围节点实现从一个范围对应至另一个范围
查看>>
Node-RED中实现HTML表单提交和获取提交的内容
查看>>
Node-RED中将CSV数据写入txt文件并从文件中读取解析数据
查看>>
Node-RED中建立TCP服务端和客户端
查看>>
Node-RED中建立Websocket客户端连接
查看>>
Node-RED中建立静态网页和动态网页内容
查看>>
Node-RED中解析高德地图天气api的json数据显示天气仪表盘
查看>>
Node-RED中连接Mysql数据库并实现增删改查的操作
查看>>
Node-RED中通过node-red-ui-webcam节点实现访问摄像头并截取照片预览
查看>>
Node-RED中配置周期性执行、指定时间阶段执行、指定时间执行事件
查看>>
Node-RED安装图形化节点dashboard实现订阅mqtt主题并在仪表盘中显示温度
查看>>
Node-RED怎样导出导入流程为json文件
查看>>
Node-RED订阅MQTT主题并调试数据
查看>>
Node-RED通过npm安装的方式对应卸载
查看>>
node-request模块
查看>>
node-static 任意文件读取漏洞复现(CVE-2023-26111)
查看>>
Node.js 8 中的 util.promisify的详解
查看>>
node.js debug在webstrom工具
查看>>
Node.js RESTful API如何使用?
查看>>
node.js url模块
查看>>