現有 40 個異步請求須要發送,但因爲某些緣由,咱們必須將同一時刻併發請求數量控制在 6 個之內,同時還要儘量快速的拿到響應結果。應該怎麼作?面試
這個問題與一道經典面試題很相似:數組
實現一個批量請求函數 multiRequest(urls, maxNum),要求以下: • 要求最大併發數 maxNum • 每當有一個請求返回,就留下一個空位,能夠增長新的請求 • 全部請求完成後,結果按照 urls 裏面的順序依次打出
串行是一個 http 請求成功後再次發起下一個 http 請求;promise
優勢瀏覽器
缺點:併發
const p = () => { return new Promise((resolve, reject) => { setTimeout(() => { console.log("1000"); resolve(); }, 1000); }); }; const p1 = () => { return new Promise((resolve, reject) => { setTimeout(() => { console.log("2000"); resolve(); }, 2000); }); }; const p2 = function () { return new Promise((resolve, reject) => { setTimeout(() => { console.log("3000"); resolve(); }, 3000); }); }; p().then(() => { return p1(); }).then(() => { return p2(); }).then(() => { console.log("end"); });
參數數組中全部 promise 都達到resolve
狀態,才執行then
回調。異步
缺點:函數
const promises = () => { return [1000, 2000, 3000].map((current) => { return new Promise((resolve, reject) => { setTimeout(() => { console.log(current); resolve(); }, current); }); }); }; Promise.all(promises()).then(() => { console.log("end"); });
Promise.all 併發限制指的是:每一個時刻併發執行的promise
數量是固定的,最終的執行結果仍是保持與原來的Promise.all
一致。fetch
添加最大併發數 maxRequestNum,全部請求完成後再返回。url
優勢:code
缺點:
const multiRequest = (fetch, params = [], maxRequestNum = 6) => { const paramsLength = params.length; let result = new Array(paramsLength).fill(false); let sendCount = 0; let finishCount = 0; return new Promise((resolve) => { while (sendCount < maxRequestNum && sendCount < paramsLength) { next(); } function handleResult(current, res) { finishCount ++; result[current] = res; if (sendCount < paramsLength) { next(); } if (finishCount >= paramsLength) { resolve(result); } } function next() { let current = sendCount++; const param = params[current]; fetch(param).then((res) => { handleResult(current, res) }).catch((err) => { handleResult(current, err) }); } }); }