摘要:在這種情況下,馬上可以想到的一個(gè)方法是我們可以使用隊(duì)列的數(shù)據(jù)結(jié)構(gòu)來實(shí)現(xiàn)調(diào)度粒度為的方法。
實(shí)現(xiàn)一個(gè)并發(fā)數(shù)可變的 Promise.all 靜態(tài)方法
Promise.all (iterable): The all function returns a new promise which is fulfilled with an array of fulfillment values for the passed promises, or rejects with the reason of the first passed promise that rejects. It resolves all elements of the passed iterable to promises as it runs this algorithm.
Promise.all 靜態(tài)方法具有如下特性:
接收一個(gè) Promise 實(shí)例的數(shù)組或具有 Iterator 接口的對象
如果元素不是 Promise 對象則使用 Promise.resolve 轉(zhuǎn)成 Promise 對象
如果全部成功狀態(tài)變?yōu)?resolved,返回值將有序組成一個(gè)數(shù)組傳給回調(diào)
只要有一個(gè)失敗狀態(tài)就變?yōu)?rejected,返回值將直接傳遞給回調(diào)
該方法的返回值是一個(gè)新的 Promise 對象
下面是 Promise.all 的簡單用法:
const p1 = Promise.resolve(1); const p2 = Promise.resolve(2); const p3 = Promise.resolve(3); Promise.all([p1, p2, p3]) .then((results) => { console.log(results); // [1, 2, 3] });
const p1 = Promise.resolve(1); const p2 = Promise.reject(2); const p3 = Promise.resolve(3); Promise.all([p1, p2, p3]) .then((results) => { console.log(results); }).catch((e) => { console.log(e); // 2 });如何實(shí)現(xiàn) Promise.all 靜態(tài)方法
Promise.all = Promise.all || function(promises) { // 如果實(shí)參不是數(shù)組則報(bào)錯(cuò)返回 if (!isArray(promises)) { throw new TypeError("You must pass an array to all."); } // 結(jié)果返回一個(gè)新的 Promise 實(shí)例 return new Promise(function(resolve, reject) { var i = 0, result = [], len = promises.length, count = len // 使用閉包記錄數(shù)組執(zhí)行順序 function resolver(index) { return function(value) { resolveAll(index, value); }; } // 只要有一個(gè)失敗狀態(tài)就變?yōu)?rejected function rejecter(reason) { reject(reason); } // 如果全部成功狀態(tài)變?yōu)?resolved function resolveAll(index, value) { result[index] = value; if (--count == 0) { resolve(result) } } // 遍歷數(shù)組并發(fā)執(zhí)行異步代碼 for (; i < len; i++) { promises[i].then(resolver(i), rejecter); } }); }實(shí)現(xiàn)一個(gè)調(diào)度粒度可變的 Promise.all 靜態(tài)方法
那么回到題目的問題:如何實(shí)現(xiàn)一個(gè)調(diào)度粒度可變的 Promise.all 靜態(tài)方法呢?這里首先可能會產(chǎn)生一個(gè)疑問就是什么叫調(diào)度粒度可變,實(shí)際上很簡單:就是給 all 方法增加一個(gè)正整數(shù)類型的參數(shù),用來標(biāo)識傳入的 Promise 實(shí)例數(shù)組中可以并發(fā)執(zhí)行的最大個(gè)數(shù)。
舉例如下,聲明三個(gè)不同異步時(shí)間的 Promise 實(shí)例并調(diào)用 all 方法,正常情況下(我們粗暴的認(rèn)為)其執(zhí)行時(shí)間應(yīng)該為200ms(當(dāng)然這是一個(gè)錯(cuò)誤的答案,在這個(gè)例子當(dāng)中我們僅考慮并發(fā)邏輯),而在調(diào)度粒度為1的 all 方法中其執(zhí)行時(shí)間應(yīng)該為450ms,在調(diào)度粒度為2的 all 方法中其執(zhí)行時(shí)間應(yīng)該為250ms,以此類推。
const p1 = new Promise((resolve) => setTimeout(() => resolve(1), 150)); const p2 = new Promise((resolve) => setTimeout(() => resolve(2), 200)); const p3 = new Promise((resolve) => setTimeout(() => resolve(3), 100)); Promise.all([p1, p2, p3]).then((r) => console.log(r));
在這種情況下,馬上可以想到的一個(gè)方法是:我們可以使用隊(duì)列的數(shù)據(jù)結(jié)構(gòu)來實(shí)現(xiàn)調(diào)度粒度為1的 all 方法。
Promise.all1 = (promises) => { if (!isArray(promises)) { throw new TypeError("You must pass an array to all."); } return new Promise((resolve, reject) => { const _q = [...promises]; const result = []; function resolver(value) { result.push(value); next(); } function rejecter(reason) { reject(reason); } function next() { if (_q.length) { _q.shift().then(resolver, rejecter); } else { resolve(result); } } next(); }); }
寫到這兒不難發(fā)現(xiàn),不同調(diào)度粒度實(shí)際上是對隊(duì)列每次推出的 Promise 實(shí)例數(shù)量最大值的約束,以及對返回結(jié)果的順序索引作出緩存,那么把代碼進(jìn)行簡單的修改即可實(shí)現(xiàn)預(yù)期的功能。
Promise.all2 = (promises, concurrent = promises.length) => { if (!Array.isArray(promises)) { throw new TypeError("You must pass an array to all."); } if (concurrent < 1) { return Promise.reject(); } return new Promise((resolve, reject) => { const queue = [...promises]; const result = []; let total = promises.length; let count = concurrent; let index = 0; function resolver(index) { return function(value) { resolveAll(index, value); }; } function resolveAll(index, value) { result[index] = value; count++; total--; next(); if (!total) { resolve(result); } } function rejecter(reason) { reject(reason); } function next() { while (queue.length && count > 0) { count--; (queue.shift())().then(resolver(index++), rejecter); } } next(); }); };
當(dāng)然這里有一個(gè)比較吊詭的地方!如果我們按照上面的代碼進(jìn)行 Promise 實(shí)例的聲明,那么在執(zhí)行到 all 方法之前它們就已經(jīng)在并發(fā)執(zhí)行,也就不會有“不同調(diào)度粒度”之說,所以為了實(shí)現(xiàn)我們預(yù)期的功能需要在 all 方法內(nèi)部把這些 Promise 實(shí)例初始化出來。
function promiseFactory(fn) { return function() { return new Promise(fn); }; } const p1 = promiseFactory((resolve) => setTimeout(() => resolve(1), 1500)); const p2 = promiseFactory((resolve) => setTimeout(() => resolve(2), 2000)); const p3 = promiseFactory((resolve) => setTimeout(() => resolve(3), 1000)); console.time("hello world!"); Promise.all2([p1, p2, p3], 1).then((r) => { console.log(r) console.timeEnd("hello world!"); // 4500+ms打印結(jié)果 }); Promise.all2([p1, p2, p3], 2).then((r) => { console.log(r) console.timeEnd("hello world!"); // 2500+ms打印結(jié)果 }); Promise.all2([p1, p2, p3]).then((r) => { console.log(r) console.timeEnd("hello world!"); // 2000+ms打印結(jié)果 });
為了能看出明顯的區(qū)別我們把定時(shí)器的時(shí)間延長到秒,執(zhí)行代碼后完美驗(yàn)證。
至此結(jié)束。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/94333.html
摘要:并發(fā)限制指的是,每個(gè)時(shí)刻并發(fā)執(zhí)行的數(shù)量是固定的,最終的執(zhí)行結(jié)果還是保持與原來的一致。換句話說,就是把生成數(shù)組的控制權(quán),交給并發(fā)控制邏輯。 背景 通常,我們在需要保證代碼在多個(gè)異步處理之后執(zhí)行,會用到: Promise.all(promises: []).then(fun: function); Promise.all可以保證,promises數(shù)組中所有promise對象都達(dá)到resol...
摘要:最近在寫一個(gè)程序,功能是下載頁面上的資源,首先拿到頁面資源鏈接列表,如要求是資源并行下載,所有資源下載結(jié)束后通知,收集錯(cuò)誤的下載鏈接。如果把上面模擬請求的注釋去掉,還會發(fā)現(xiàn)是在結(jié)束后就執(zhí)行了,而后面的請求還未結(jié)束。 最近在寫一個(gè)Node.js程序,功能是下載頁面上的資源,首先拿到頁面資源鏈接列表,如: [ https://xxx.com/img/logo.jpg, https:...
摘要:第二種則一定會執(zhí)行所有的異步函數(shù),即便你需要使用的是這些高階函數(shù)。并發(fā)實(shí)現(xiàn)的異步數(shù)組然后修改,使用即可上面的其他內(nèi)容終結(jié)整個(gè)鏈?zhǔn)讲僮鞑⒎祷亟Y(jié)果這里使用是為了兼容的調(diào)用方式調(diào)用方式不變。 JavaScript 異步數(shù)組 吾輩的博客原文: https://blog.rxliuli.com/p/5e... 場景 吾輩是一只在飛向太陽的螢火蟲 JavaScript 中的數(shù)組是一個(gè)相當(dāng)泛用性的...
摘要:本次的任務(wù)假如。。。。。引擎發(fā)生了重大故障,方法變成了,為了拯救世界,需要開發(fā)一個(gè)模塊來解決此問題。實(shí)現(xiàn)首先要知道是什么是對異步編程的一種抽象。數(shù)組中任何一個(gè)為的話,則整個(gè)調(diào)用會立即終止,并返回一個(gè)的新的對象。 本次的任務(wù) 假如。。。。。 JavaScript v8 引擎發(fā)生了重大故障,Promise.all 方法變成了 undefined ,為了拯救 JavaScript 世界,需要...
摘要:并發(fā)模塊本身有兩種不同的類型進(jìn)程和線程,兩個(gè)基本的執(zhí)行單元。調(diào)用以啟動新線程。在大多數(shù)系統(tǒng)中,時(shí)間片發(fā)生不可預(yù)知的和非確定性的,這意味著線程可能隨時(shí)暫?;蚧謴?fù)。 大綱 什么是并發(fā)編程?進(jìn)程,線程和時(shí)間片交織和競爭條件線程安全 策略1:監(jiān)禁 策略2:不可變性 策略3:使用線程安全數(shù)據(jù)類型 策略4:鎖定和同步 如何做安全論證總結(jié) 什么是并發(fā)編程? 并發(fā)并發(fā)性:多個(gè)計(jì)算同時(shí)發(fā)生。 在現(xiàn)代...
閱讀 2302·2021-11-24 09:38
閱讀 2166·2021-11-22 14:44
閱讀 1162·2021-07-29 13:48
閱讀 2622·2019-08-29 13:20
閱讀 1123·2019-08-29 11:08
閱讀 2065·2019-08-26 10:58
閱讀 1270·2019-08-26 10:55
閱讀 3165·2019-08-26 10:39