您现在的位置:新闻首页>资本 > 了解Promise中的all()、race()、allSettled()方法
了解Promise中的all()、race()、allSettled()方法
从ES6 开始,我们大都使用的是 Promise.all()和Promise.race(),Promise.allSettled() 提案已经到第4阶段,因此将会成为ECMAScript 2020的一部分。
1.概述
Promise.all(promises: Iterable<Promise>): Promise<Array>
Promise.all(iterable) 方法返回一个 Promise 实例,此实例在 iterable 参数内所有的 promise 都“完成(resolved)”或参数中不包含 promise 时回调完成(resolve);如果参数中 promise 有一个失败(rejected),此实例回调失败(reject),失败原因的是第一个失败 promise 的结果
Promise.race(promises: Iterable<Promise>): Promise
Promise.race(iterable) 方法返回一个 promise,一旦迭代器中的某个promise解决或拒绝,返回的 promise就会解决或拒绝。
Promise.allSettled(promises: Iterable<Promise>): Promise<Array<SettlementObject>>
Promise.allSettled()方法返回一个promise,该promise在所有给定的promise已被解析或被拒绝后解析,并且每个对象都描述每个promise的结果。2. 回顾: Promise 状态
给定一个返回Promise的异步操作,以下这些是Promise的可能状态:
pending: 初始状态,既不是成功,也不是失败状态。fulfilled: 意味着操作成功完成。rejected: 意味着操作失败。Settled: Promise要么被完成,要么被拒绝。Promise一旦达成,它的状态就不再改变。
3.什么是组合
又称部分-整体模式,将对象整合成树形结构以表示“部分整体”的层次结构。组合模式使得用户对单个对象和组合对象的使用具有一致性,它基于两种函数:
基元函数(简短:基元)创建原子块。组合函数(简称:组合)将原子和/或复合件组合在一起以形成复合件。
对于 JS 的 Promises 来说
基元函数包括:Promise.resolve()、Promise.reject()组合函数:Promise.all(), Promise.race(), Promise.allSettled()4. Promise.all()
Promise.all()的类型签名:
Promise.all(promises: Iterable<Promise>): Promise<Array>
返回情况:
完成(Fulfillment):
如果传入的可迭代对象为空,Promise.all 会同步地返回一个已完成(resolved)状态的promise。
如果所有传入的 promise 都变为完成状态,或者传入的可迭代对象内没有 promise,Promise.all 返回的 promise 异步地变为完成。
在任何情况下,Promise.all 返回的 promise 的完成状态的结果都是一个数组,它包含所有的传入迭代参数对象的值(也包括非 promise 值)。
失败/拒绝(Rejection):
如果传入的 promise 中有一个失败(rejected),Promise.all 异步地将失败的那个结果给失败状态的回调函数,而不管其它 promise 是否完成。
来个例子:
const promises = [ Promise.resolve('a'), Promise.resolve('b'), Promise.resolve('c'), ]; Promise.all(promises) .then((arr) => assert.deepEqual( arr, ['a', 'b', 'c'] ));
如果其中的一个 promise 被拒绝,那么又是什么情况:
const promises = [ Promise.resolve('a'), Promise.resolve('b'), Promise.reject('ERROR'), ]; Promise.all(promises) .catch((err) => assert.equal( err, 'ERROR' ));
下图说明Promise.all()是如何工作的
4.1 异步 .map() 与 Promise.all()
数组转换方法,如.map()、.filter()等,用于同步计算。例如
function timesTwoSync(x) { return 2 * x; } const arr = [1, 2, 3]; const result = arr.map(timesTwoSync); assert.deepEqual(result, [2, 4, 6]);
如果.map()的回调是基于Promise的函数会发生什么? 使用这种方式 .map()返回的的结果是一个Promises数组。
Promises数组不是普通代码可以使用的数据,但我们可以通过Promise.all()来解决这个问题:它将Promises数组转换为Promise,并使用一组普通值数组来实现。
function timesTwoAsync(x) { return new Promise(resolve => resolve(x * 2)); } const arr = [1, 2, 3]; const promiseArr = arr.map(timesTwoAsync); Promise.all(promiseArr) .then(result => { assert.deepEqual(result, [2, 4, 6]); });更实际工作上关于 .map()示例
接下来,咱们使用.map()和Promise.all()从Web下载文件。 首先,咱们需要以下帮助函数:
function downloadText(url) { return fetch(url) .then((response) => { // (A) if (!response.ok) { // (B) throw new Error(response.statusText); } return response.text(); // (C) }); }
downloadText()使用基于Promise的fetch API 以字符串流的方式下载文件:
首先,它异步检索响应(第A行)。response.ok(B行)检查是否存在“找不到文件”等错误。如果没有错误,使用.text()(第C行)以字符串的形式取回文件的内容。
在下面的示例中,咱们 下载了两个文件
const urls = [ 'Promise.all()的一个简版实现 function all(iterable) { return new Promise((resolve, reject) => { let index = 0; for (const promise of iterable) { // Capture the current value of `index` const currentIndex = index; promise.then( (value) => { if (anErrorOccurred) return; result[currentIndex] = value; elementCount++; if (elementCount === result.length) { resolve(result); } }, (err) => { if (anErrorOccurred) return; anErrorOccurred = true; reject(err); }); index++; } if (index === 0) { resolve([]); return; } let elementCount = 0; let anErrorOccurred = false; const result = new Array(index); }); }5. Promise.race()
Promise.race()方法的定义:
Promise.race(promises: Iterable<Promise>): Promise
Promise.race(iterable) 方法返回一个 promise,一旦迭代器中的某个promise解决或拒绝,返回的 promise就会解决或拒绝。来几个例子,瞧瞧:
const promises = [ new Promise((resolve, reject) => setTimeout(() => resolve('result'), 100)), // (A) new Promise((resolve, reject) => setTimeout(() => reject('ERROR'), 200)), // (B) ]; Promise.race(promises) .then((result) => assert.equal( // (C) result, 'result'));
在第 A 行,Promise 是完成状态 ,所以 第 C 行会执行(尽管第 B 行被拒绝)。
如果 Promise 被拒绝首先执行,在来看看情况是嘛样的:
const promises = [ new Promise((resolve, reject) => setTimeout(() => resolve('result'), 200)), new Promise((resolve, reject) => setTimeout(() => reject('ERROR'), 100)), ]; Promise.race(promises) .then( (result) => assert.fail(), (err) => assert.equal( err, 'ERROR'));
注意,由于 Promse 先被拒绝,所以 Promise.race() 返回的是一个被拒绝的 Promise
这意味着Promise.race([])的结果永远不会完成。
下图演示了Promise.race()的工作原理:
Promise.race() 在 Promise 超时下的情况
在本节中,我们将使用Promise.race()来处理超时的 Promise。 以下辅助函数:
function resolveAfter(ms, value=undefined) { return new Promise((resolve, reject) => { setTimeout(() => resolve(value), ms); }); }
resolveAfter() 主要做的是在指定的时间内,返回一个状态为 resolve 的 Promise,值为为传入的 value
调用上面方法:
function timeout(timeoutInMs, promise) { return Promise.race([ promise, resolveAfter(timeoutInMs, Promise.reject(new Error('Operation timed out'))), ]); }
timeout() 返回一个Promise,该 Promise 的状态取决于传入 promise 状态 。
其中 timeout 函数中的 resolveAfter(timeoutInMs, Promise.reject(new Error('Operation timed out')) ,通过 resolveAfter 定义可知,该结果返回的是一个被拒绝状态的 Promise。
再来看看timeout(timeoutInMs, promise)的运行情况。如果传入promise在指定的时间之前状态为完成时,timeout 返回结果就是一个完成状态的 Promise,可以通过.then的第一个回调参数处理返回的结果。
timeout(200, resolveAfter(100, 'Result!')) .then(result => assert.equal(result, 'Result!'));
相反,如果是在指定的时间之后完成,刚 timeout 返回结果就是一个拒绝状态的 Promise,从而触发catch方法指定的回调函数。
timeout(100, resolveAfter(2000, 'Result!')) .catch(err => assert.deepEqual(err, new Error('Operation timed out')));
重要的是要了解“Promise 超时”的真正含义:
如果传入入Promise 较到的得到解决,其结果就会给返回的 Promise。如果没有足够快得到解决,输出的 Promise 的状态为拒绝。
也就是说,超时只会阻止传入的Promise,影响输出 Promise(因为Promise只能解决一次), 但它并没有阻止传入Promise的异步操作。
5.2 Promise.race() 的一个简版实现
以下是 Promise.race()的一个简化实现(它不执行安全检查)
function race(iterable) { return new Promise((resolve, reject) => { for (const promise of iterable) { promise.then( (value) => { if (settlementOccurred) return; settlementOccurred = true; resolve(value); }, (err) => { if (settlementOccurred) return; settlementOccurred = true; reject(err); }); } let settlementOccurred = false; }); }6.Promise.allSettled()
“Promise.allSettled”这一特性是由Jason Williams,Robert Pamely和Mathias Bynens提出。
promise.allsettle()方法的定义:
Promise.allSettled(promises: Iterable<Promise>)
: Promise<Array<SettlementObject>>
它返回一个Array的Promise,其元素具有以下类型特征:
type SettlementObject<T> = FulfillmentObject<T> RejectionObject; interface FulfillmentObject<T> { status: 'fulfilled'; value: T; } interface RejectionObject { status: 'rejected'; reason: unknown; }
Promise.allSettled()方法返回一个promise,该promise在所有给定的promise已被解析或被拒绝后解析,并且每个对象都描述每个promise的结果。
举例说明, 比如各位用户在页面上面同时填了3个独立的表单, 这三个表单分三个接口提交到后端, 三个接口独立, 没有顺序依赖, 这个时候我们需要等到请求全部完成后给与用户提示表单提交的情况
在多个promise同时进行时咱们很快会想到使用Promise.all来进行包装, 但是由于Promise.all的短路特性, 三个提交中若前面任意一个提交失败, 则后面的表单也不会进行提交了, 这就与咱们需求不符合.
Promise.allSettled跟Promise.all类似, 其参数接受一个Promise的数组, 返回一个新的Promise, 唯一的不同在于, 其不会进行短路, 也就是说当Promise全部处理完成后我们可以拿到每个Promise的状态, 而不管其是否处理成功.
下图说明promise.allsettle()是如何工作的
6.1 Promise.allSettled() 例子
这是Promise.allSettled() 使用方式快速演示示例
Promise.allSettled([ Promise.resolve('a'), Promise.reject('b'), ]) .then(arr => assert.deepEqual(arr, [ { status: 'fulfilled', value: 'a' }, { status: 'rejected', reason: 'b' }, ]));6.2 Promise.allSettled() 较复杂点的例子
这个示例类似于.map()和Promise.all()示例(我们从其中借用了downloadText()函数):我们下载多个文本文件,这些文件的url存储在一个数组中。但是,这一次,咱们不希望在出现错误时停止,而是希望继续执行。Promise.allSettled()允许咱们这样做:
const urls = [ '6.3 Promise.allSettled() 的简化实现
这是promise.allsettle()的简化实现(不执行安全检查)
function allSettled(iterable) { return new Promise((resolve, reject) => { function addElementToResult(i, elem) { result[i] = elem; elementCount++; if (elementCount === result.length) { resolve(result); } } let index = 0; for (const promise of iterable) { // Capture the current value of `index` const currentIndex = index; promise.then( (value) => addElementToResult( currentIndex, { status: 'fulfilled', value }), (reason) => addElementToResult( currentIndex, { status: 'rejected', reason })); index++; } if (index === 0) { resolve([]); return; } let elementCount = 0; const result = new Array(index); }); }7. 短路特性
Promise.all() 和 romise.race() 都具有 短路特性
Promise.all(): 如果参数中 promise 有一个失败(rejected),此实例回调失败(reject)
Promise.race():如果参数中某个promise解决或拒绝,返回的 promise就会解决或拒绝。
8.并发性和 Promise.all()8.1 顺序执行与并发执行
考虑下面的代码:
asyncFunc1() .then(result1 => { assert.equal(result1, 'one'); return asyncFunc2(); }) .then(result2 => { assert.equal(result2, 'two'); });
使用.then()顺序执行基于Promise的函数:只有在 asyncFunc1()的结果被解决后才会执行asyncFunc2() 。
而 Promise.all() 是并发执行的
Promise.all([asyncFunc1(), asyncFunc2()]) .then(arr => { assert.deepEqual(arr, ['one', 'two']); });8.2 并发技巧:关注操作何时开始
确定并发异步代码的技巧:关注异步操作何时启动,而不是如何处理它们的Promises。
例如,下面的每个函数都同时执行asyncFunc1()和asyncFunc2(),因为它们几乎同时启动。
function concurrentAll() { return Promise.all([asyncFunc1(), asyncFunc2()]); } function concurrentThen() { const p1 = asyncFunc1(); const p2 = asyncFunc2(); return p1.then(r1 => p2.then(r2 => [r1, r2])); }
另一方面,以下两个函数依次执行asyncFunc1()和asyncFunc2(): asyncFunc2()仅在asyncFunc1()的解决之后才调用。
function sequentialThen() { return asyncFunc1() .then(r1 => asyncFunc2() .then(r2 => [r1, r2])); } function sequentialAll() { const p1 = asyncFunc1(); const p2 = p1.then(() => asyncFunc2()); return Promise.all([p1, p2]); }8.3 Promise.all() 与 Fork-Join 分治编程
Promise.all() 与并发模式“fork join”松散相关。重温一下咱们前面的一个例子:
Promise.all([ // (A) fork downloadText('Fork:在A行中,分割两个异步任务并同时执行它们。Join:在B行中,对每个小任务得到的结果进行汇总。
英文原文:
编程教学!!
下一篇:一个汉字的机内码需要几个字节储存
-
经济 业界 推荐 美圆指数 29美元 福汇外汇 港币兑换美元 公信宝 币世界 ok币 加拿大元汇率 金条价格走势 ok交易所 白银套利 ppi指数 金价走势分析 中币交易所 玩客币行情 港币兑美元 马来西亚货币 今日复明日 旧日噩梦 bullish 海曼明斯基 绿天鹅 黄金行情走势 汇率日元 火币pro 莱茨狗 fx57 美元价格 币世界快讯 金价格走势图 隔夜利率 全球货币战争 波场tron 2199美元 stdaily 伊朗油价 国际石油行情 btcchina 美元日元汇率 恒生指数实时 大立光股票 回升 hc币 夏盈盈 希腊公投 市场黄金价格 黄金k线走势图 蜡烛图 单均线交易 日元美元 国际油价趋势 比特币白皮书 2012年金价走势 usdt 白银价钱 今日石油价格 fx1800 缩表 油价走势 台股 sdag 杨林科 港币汇率 明斯基时刻 猛烈打压 stellar 隔夜美股行情 白银行情 dp1s 油价 微比特 meiyuan 香港恒生指数 成交量分析 白银比例 实时行情 白银 国际石油 ltc是什么币种 美元指数走势 期货实时行情 美元兑澳元 中期选举 美元指数dini rsi指标 美金兑港币 谦益农业 硬币回收价表 今天美元走势 太一云 间谍车 加元汇率 国际石油价格 意大利国债 澳元走势预测 btc挖矿 美原油行情 即时外汇 制造业指数 澳元汇率 美国股市休市 下周美元走势 欧债 玩客云 美原油连 道琼指数 币种 美元汇率走势 文章档案 外汇止损多少 以太 挖矿 vshen 极路由hiwifi 汇丰pmi adx 美元兑日元 全球央行年会 btm 空投 安币交易所 chaobi otc交易平台 金价 标普500期货 加币汇率走势 日元兑换美元 伦敦铜价 著名财经 国际油价查询 etc 外汇学习 美债收益率 阿希币 pEE币 什么是头寸 纽交所 钻石底 德国30 799澳元 持仓报告 玩客 原油走势图 港股恒生指数 欧元下跌 420欧元 金子价格 加元走势图 1.11111E+11 xrp 美元指数k线图 金价走势预测 最新黄金价格 铜价格走势图 黄金降价 汇率欧元 金针探底 原油成本 美元 strllar 泰奇猫 圈牌 金价走势 以太币 lme铜实时行情 eos价格走势 欧元兑美金 外汇基本知识 联邦基金利率 伦敦银走势图 基本面分析 空头回补 云鱼 py6是什么货币 rsi指标详解 265万澳元 国际油价格 gateio wti原油走势图 门罗币 白银价格走势 欧盟财长会议 外汇咨询 交叉盘 外汇初学 房价指数 cbt 比特股 ltc 隐私政策 石油危机 日圆汇率 英国股市指数 原油最新价格 行情报价 自动减支 黄金市场价 全球指数 imtoken 币投资 10美金 eos币价格 相对强弱指标 黄金年走势图 美原油 加元美元 虚拟币 值多少钱 国际油价 外汇哈里森 外汇交易分析 白银价格分析 日bi btcc 标准普尔500 wti原油价格 zbcom 和币 度宇宙 技术指标分析 全球股市指数 币久 白银价格趋势 克龙 银行回收硬币 hiwifi 贝尔链 美元兑换欧元 后座议员 黄金市场行情 德拉基讲话 UES 道琼斯k线图 美元对日元 k线图分析 恒生指数 英国脱欧时间 港股指数 比特币之父 bin 今日原油 jinjia 日经225指数 比特币价格 英镑汇率 742 大立光 外汇走势 上吊线 趣步APP被调查 肖野 理财三 铜走势图 艾达 吞阳 coinex 欧元美金 赵长鹏 法郎汇率 9g游戏 英国脱欧结果 硅谷bbs 俄罗斯火星人 铜价 什么叫头寸
-
关于黄金交易 放学 仅数 k线图解读 btc行情 招生 欧盟会议 英格兰银行 俄罗斯火星人 dragon 欧美黄金 513部队 老赖 美国总统 姐姐 隔夜利率 引而不发 顾比均线 ok交易所 化工 灵魂 退出 1199元 加拿大元汇率 2599澳元 投资性需求 比特币交易 sleepfx失效 货币狼烟 日内交易者 即时外汇牌价 台湾股票查询 持仓报告 国际油价格 提问 再添 消防员 以为 提高 慈善 青年 汽车业 遭遇 好酒 经济 黄金k线走势图 第六计 美国原油 xunleiyun 港元兑美元 蝴蝶币 纽币汇率 非美货币 港币美元汇率 港币兑换 白银对冲套利 日经225 美伊关系 标准普尔500 吞阳 2012年金价走势 微博 人说 一层 联手 火锅 餐饮 揭秘 5 质量 总决赛 开业 骑行 效率 兑美元 什么是头寸 黄金价格分析 早晨之星 印度卢比 bullish 传产 瑞波币 账户银走势图 coinegg 27亿美金 铜期货 链克行情 法郎汇率 币投资 欧元走势 欧元美金 相对强弱指标 恒生指数 欧元兑美金 美元指数k线图 美元兑澳元 杨林科 hc币 间谍车 谦益农业 播控云 多好 侧脸 热情 猫日 fashion 欧洲 品牌 尽在 专业 考场 顺序 低帮版 秦岭 突尼斯 多门 攻坚战 取消 由于 可握 地震 客户端 高考 业界 下周美元走势 国际石油 以太坊 绿天鹅 ppi指数 外汇初学 白银价格趋势 欧盟财长会议 港股实时行情 空头行情 黄金走势 白银行情 黄金的价格 威比特 石油走势 南非兰特 德国股票指数 道指指数 黄金市场行情 外汇走势 美元对日元 imtoken 铜走势图 港股指数 比特币价格 硅谷bbs 英国脱欧时间 外汇交易分析 美元兑日元 制造业指数 即时外汇 虚拟币 成交量分析 云鱼 油价走势 最新黄金价格 国际石油价格 sdag 以太 otc交易平台 阿希币 ok币 军事 热线 违规 何雷 陆战队 沦丧 拍卖 靓号 最高 妻子 伤残 欧冠 传销 两男 打印机 2018 一下 摄像头 三星 依法 精神 房屋 家装 更好 匹克 在的 半裙 闪耀 来源 连接 至少 信 经贸 又讲 40个 依然 电池 大规模 中国特色 好不好 户型 行业 体育用品 今天美元走势 白银 rsi指标 白银比例 房价指数 什么是比特币 黄金行情走势 外汇学堂 fx外汇 著名财经 币热度 最新国际油价 英镑兑欧元 塞浦路斯危机 财经要闻 eunice 一级市场 莱特币矿池 俄罗斯物价 美金兑换港币 菲律宾比索 pee 508888 加拿大元走势 美国油价 bitebi 马来西亚币 道琼斯工业 今日美股 赵长鹏 btctrade 上吊线 欧洲峰会 肖野 coinex 英镑汇率 eos价格走势 度宇宙 eos币价格 理财三 基本面分析 联邦基金利率 交叉盘 9g游戏 英国脱欧结果 日bi 金价走势预测 hiwifi 美原油连 10美金 克龙 比特币白皮书 艾达 币种 美元价格 中期选举 德拉基讲话 btc挖矿 全球货币战争
-
朋友 ok交易所 港币兑美元 业界 加拿大元汇率 美元兑日元 dp1s 513部队 绿天鹅 俄罗斯火星人 恒生指数实时 仅数 海曼明斯基 29美元 2012年金价走势 推荐 2199美元 汇率日元 美元指数k线图 火锅 极路由hiwifi 币世界 明细 印度卢比 美圆指数 ok币 经济 隔夜利率 金条价格走势 玩客币行情 火币pro 攻坚战 1199元 黄金k线走势图 关于黄金交易 即时外汇牌价 币世界快讯 有限公司 回升 btc挖矿 钻石底 你吗 发表声明 stdaily 金价格走势图 港币兑换美元 杨林科 下周美元走势 金价走势分析 福汇外汇 12334 持仓报告 宽松 首登 日本 市场黄金价格 fx1800 外汇趋势分析 国际石油价格 玩客云 老赖 白银价格趋势 19929日元 原油最新价格 油价走势 meiyuan 间谍车 罪证 鲜 效率 美原油行情 蜡烛图 意大利国债 fx 缩表 道琼指数 空投 微比特 3号 3人 创造 体育用品 bullish 比特币挖矿机 门罗币 香港恒生指数 美元日元汇率 谦益农业 莱茨狗 突尼斯 美国国债利率 w底形态 日圆汇率 中币交易所 安币交易所 当天 永久 成鬼 fashion 处去 匹克 财务省 白银 今日石油价格 瑞士货币 火币比特币 铜价 zbcom coinex 圈牌 美原油连 比特币白皮书 最新黄金价格 usdt hc币 otc交易平台 军事 强国 胳膊 交警 对比 核实 查询 今天美元走势 房价指数 著名财经 欧债 今日恒生指数 白银价格分析 后座议员 外汇走势 btcc 基本面分析 欧元兑美金 外汇止损多少 外汇学习 英国股市指数 币种 全球货币战争 金价 日元美元 完全 靓号 驾校 29日 大立光股票 明斯基时刻 港币汇率 白银价钱 2599澳元 g7集团 拜登大胜 黄金市场行情 全球股市指数 黄金年走势图 vshen cbt 国际油价格 美元指数走势 40年 草色 用户 发布 顺序 慈善 着力点 又放 黄金行情走势 白银行情 国际石油行情 伊朗油价 股市行情图 黄金etf是什么 wti原油价格 联邦基金利率 英国脱欧时间 pEE币 台股 btcchina 硬币回收价表 美国股市休市 金子价格 驱逐舰 救援 美元 腾讯 搭载 代人 他来 普爱 国企 班主任 行情报价 隔夜美股行情 gasstation 德国30 eos币价格 伦敦银走势图 金价走势预测 hiwifi 10美金 原油走势图 以太 阿希币 热线 获赔 尽在 地方 怎样 青年 包括 国际油价 欧盟财长会议 单均线交易 汇率欧元 什么是头寸 白银套利 欧元美金 原油成本 港股恒生指数 外汇咨询 外汇交易分析 值多少钱 制造业指数 成交量分析 泰奇猫 澳元走势预测 国际油价趋势 太一云 拍卖 化工 选科分 总决赛 打我 多门 网售 千万 汇丰pmi 外汇哈里森 外汇初学 美金兑港币 白银比例 港股实时行情 国际油价查询 美元汇率走势 黄金走势 期货实时行情 今日日历 瑞士法郎汇率 k线图解读 币投资 贝尔链 265万澳元 标准普尔500 欧元下跌 恒生指数 日bi 美元指数dini 虚拟币 空头回补 澳元汇率 福州 人用 揭秘 Find 电池 猛烈打压 自动减支







