JS前端并發(fā)多個相同的請求控制為只發(fā)一個請求方式
描述如下
- 同時發(fā)多個相同的請求,如果第一個請求成功,那么剩余的請求都不會發(fā)出,成功的結(jié)果作為剩余請求返回
- 如果第一個請求失敗了,那么接著發(fā)編號為2的請求,如果請求成功,那么剩余的請求都不會發(fā)出,成功的結(jié)果作為剩余請求返回
- 如果第二個請求失敗了,那么接著發(fā)編號為3的請求,如果請求成功,那么剩余的請求都不會發(fā)出,成功的結(jié)果作為剩余請求返回
...
以此遞推,直到遇到最壞的情況需要發(fā)送最后一個請求
并發(fā): 一個接口請求還處于pending,短時間內(nèi)就發(fā)送相同的請求
async function fetchData (a) { const data = await fetch('//127.0.0.1:3000/test') const d = await data.json(); console.log(d); return d; } fetchData(2) // 編號 1 fetchData(2) // 2 fetchData(2) // 3 fetchData(2) // 4 fetchData(2) // 4 fetchData(2) // 5 fetchData(2) fetchData(2)
老版本cachedAsync
我之前使用過vue
的緩存函數(shù)緩存成功的請求, 實(shí)現(xiàn)是這樣的。下面的cachedAsync
只會緩存成功的請求,如果失敗了,直接拉起新的請求。但是如果是上面的并發(fā)場景,相同的請求因?yàn)闊o法命中緩存,會出現(xiàn)連續(xù)發(fā)送三個請求的問題,無法處理這種并發(fā)的場景。
const cachedAsync = function(fn) { const cache = Object.create(null); return async str => { const hit = cache[str]; if (hit) { return hit; } // 只緩存成功的Promise, 失敗直接重新請求 return (cache[str] = await fn(str)); }; }; const fetch2 = cachedAsync(fetchData) fetch2(2); fetch2(2); fetch2(2);
進(jìn)階版本
首先緩存是必須的,那么我們只要處理怎么控制并發(fā)即可??梢杂羞@么一個思路
- 每個請求都返回一個新的Promise, Promise的exector的執(zhí)行時機(jī),通過一個隊(duì)列保存。
- 當(dāng)隊(duì)列長度為1的時候,執(zhí)行一次請求,如果請求成功,那么遍歷隊(duì)列中的exector,拿到請求的結(jié)果然后resolve。
- 如果請求失敗了,那么就把這個Promise reject掉,同時出棧。然后遞歸調(diào)用
next
- 直到exector隊(duì)列清空為止
const cacheAsync = (promiseGenerator, symbol) => { const cache = new Map(); const never = Symbol(); return async (params) => { return new Promise((resolve, reject) => { // 可以提供鍵值 symbol = symbol || params; let cacheCfg = cache.get(symbol); if (!cacheCfg) { cacheCfg = { hit: never, exector: [{ resolve, reject }], }; cache.set(symbol, cacheCfg); } else { // 命中緩存 if (cacheCfg.hit !== never) { return resolve(cacheCfg.hit) } cacheCfg.exector.push({ resolve, reject }); } const { exector } = cacheCfg; // 處理并發(fā),在請求還處于pending過程中就發(fā)起了相同的請求 // 拿第一個請求 if (exector.length === 1) { const next = async () => { try { if (!exector.length) return; const response = await promiseGenerator(params); // 如果成功了,那么直接resolve掉剩余同樣的請求 while (exector.length) { // 清空 exector.shift().resolve(response); } // 緩存結(jié)果 cacheCfg.hit = response; } catch (error) { // 如果失敗了 那么這個promise的則為reject const { reject } = exector.shift(); reject(error); next(); // 失敗重試,降級為串行 } }; next(); } }); }; };
測試cacheAsync
需要測試的場景
- 請求接口隨機(jī)出現(xiàn)成功或者失敗
- 成功預(yù)期結(jié)果,剩余的請求都不會發(fā)出
- 失敗重試,接著發(fā)下一個請求
快速搭建一個服務(wù)器
const koa = require("koa"); const app = new koa(); function sleep(seconds) { return new Promise((resolve, reject) => { setTimeout(resolve, seconds); }); } app.use(async (ctx, next) => { if (ctx.url === "/test") { await sleep(200); const n = Math.random(); // 隨機(jī)掛掉接口 if (n > 0.8) { ctx.body = n; } else { ctx.status = 404 ctx.body = '' } next(); } }); app.listen(3000, "127.0.0.1", () => console.log("listening on 127.0.0.1:3000") );
客戶端
var fetch2 = cacheAsync(fetchData, "test2"); async function fetchData(a) { const data = await fetch("http://127.0.0.1:3000/test"); const d = await data.json(); console.log(d); return d; } // 并發(fā)6個相同的請求 console.log(fetch2(2)); console.log(fetch2(2)); console.log(fetch2(2)); console.log(fetch2(2)); console.log(fetch2(2)); console.log(fetch2(2));
看下測試結(jié)果,刷新下頁面
第一次運(yùn)氣很好,第一次接口就請求成功,只發(fā)送了一個請求
第二次測試運(yùn)氣不好,最后一個請求才成功,也是最差的場景
第三次測試,請求第三次成功了
測試下緩存 在控制臺主動請求fetch2
,成功命中。
從測試結(jié)果來看是正確的,符合了并發(fā)和緩存的場景。有人會問為什么要緩存接口,舉個場景。輸入關(guān)鍵字搜索,監(jiān)聽的是input事件,在你增刪關(guān)鍵字的時候,就會出現(xiàn)請求參數(shù)一樣的場景,這時候就符合防抖+前端接口緩存的方式。遇到相同關(guān)鍵字直接拉之前的緩存。
提示
這個緩存因?yàn)槭情]包的方式,因此刷新頁面緩存也失效了。不過我認(rèn)為這個是理應(yīng)如此,因?yàn)榇蟛糠謭鼍八⑿马撁?,就是要重置狀態(tài),如果要持久化,還不如保存到本地存儲。
以上就是JS前端并發(fā)多個相同的請求控制為只發(fā)一個請求的詳細(xì)內(nèi)容,更多關(guān)于JS并發(fā)多相同請求控制為一個的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
arco?design按需導(dǎo)入報(bào)錯排查思路與解決方案解析
這篇文章主要為大家介紹了arco?design?按需導(dǎo)入報(bào)錯排查思路與解決方案解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-03-03JavaScript中的設(shè)計(jì)模式 單例模式
這篇文章主要給大家介紹的是JavaScript中的單例模式,設(shè)計(jì)模式代表了最佳的實(shí)踐,通常被有經(jīng)驗(yàn)的面向?qū)ο蟮能浖_發(fā)人員所采用。設(shè)計(jì)模式是軟件開發(fā)人員在軟件開發(fā)過程中面臨的一般問題的解決方案,需要的朋友可以參考一下2021-09-09JS封裝轉(zhuǎn)換前后端接口數(shù)據(jù)格式工具函數(shù)下劃線<=>大寫
這篇文章主要為大家介紹了JS優(yōu)雅封裝轉(zhuǎn)換前后端接口數(shù)據(jù)格式工具函數(shù)下劃線<=>大寫實(shí)現(xiàn)詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-03-03微信小程序與php 實(shí)現(xiàn)微信支付的簡單實(shí)例
這篇文章主要介紹了微信小程序與php 實(shí)現(xiàn)微信支付的簡單實(shí)例的相關(guān)資料,需要的朋友可以參考下2017-06-06