<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
...
以此遞推,直到遇到最壞的情況需要傳送最後一個請求並行: 一個介面請求還處於pending,短時間內就傳送相同的請求
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)
我之前使用過vue
的快取函數快取成功的請求, 實現是這樣的。下面的cachedAsync
只會快取成功的請求,如果失敗了,直接拉起新的請求。但是如果是上面的並行場景,相同的請求因為無法命中快取,會出現連續傳送三個請求的問題,無法處理這種並行的場景。
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);
首先快取是必須的,那麼我們只要處理怎麼控制並行即可。可以有這麼一個思路
next
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; // 處理並行,在請求還處於pending過程中就發起了相同的請求 // 拿第一個請求 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); } // 快取結果 cacheCfg.hit = response; } catch (error) { // 如果失敗了 那麼這個promise的則為reject const { reject } = exector.shift(); reject(error); next(); // 失敗重試,降級為序列 } }; next(); } }); }; };
需要測試的場景
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(); // 隨機掛掉介面 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("//127.0.0.1:3000/test"); const d = await data.json(); console.log(d); return d; } // 並行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));
看下測試結果,重新整理下頁面
第一次運氣很好,第一次介面就請求成功,只傳送了一個請求
第二次測試運氣不好,最後一個請求才成功,也是最差的場景
第三次測試,請求第三次成功了
測試下快取 在控制檯主動請求fetch2
,成功命中。
從測試結果來看是正確的,符合了並行和快取的場景。有人會問為什麼要快取介面,舉個場景。輸入關鍵字搜尋,監聽的是input事件,在你增刪關鍵字的時候,就會出現請求引數一樣的場景,這時候就符合防抖+前端介面快取的方式。遇到相同關鍵字直接拉之前的快取。
這個快取因為是閉包的方式,因此重新整理頁面快取也失效了。不過我認為這個是理應如此,因為大部分場景重新整理頁面,就是要重置狀態,如果要持久化,還不如儲存到本地儲存。
以上就是JS前端並行多個相同的請求控制為只發一個請求的詳細內容,更多關於JS並行多相同請求控制為一個的資料請關注it145.com其它相關文章!
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45