字節面試官:如何實現Ajax并發請求控制

偷偷告訴你,點此抽獎送紅包還送3本比紅寶書還貴的書

實現一個批量請求函數 multiRequest(urls, maxNum),要求如下:
??要求最大并發數?maxNum
??每當有一個請求返回,就留下一個空位,可以增加新的請求
??所有請求完成后,結果按照?urls?里面的順序依次打出

這道字節跳動的題目我想很多同學應該都或多或少的見過,下面我會依次從出現的場景、問題的分析到最終的實現,一步步力求深入淺出的給出這道題目的完整解析。

場景

假設現在有這么一種場景:現有 30 個異步請求需要發送,但由于某些原因,我們必須將同一時刻并發請求數量控制在 5 個以內,同時還要盡可能快速的拿到響應結果。

應該怎么做?

首先我們來了解一下 Ajax的串行和并行。

基于 Promise.all 實現 Ajax 的串行和并行

我們平時都是基于promise來封裝異步請求的,這里也主要是針對異步請求來展開。

  • 串行:一個異步請求完了之后在進行下一個請求

  • 并行:多個異步請求同時進行

通過定義一些promise實例來具體演示串行/并行。

串行

var?p?=?function?()?{return?new?Promise(function?(resolve,?reject)?{setTimeout(()?=>?{console.log('1000')resolve()},?1000)})
}
var?p1?=?function?()?{return?new?Promise(function?(resolve,?reject)?{setTimeout(()?=>?{console.log('2000')resolve()},?2000)})
}
var?p2?=?function?()?{return?new?Promise(function?(resolve,?reject)?{setTimeout(()?=>?{console.log('3000')resolve()},?3000)})
}p().then(()?=>?{return?p1()
}).then(()?=>?{return?p2()
}).then(()?=>?{console.log('end')
})

如示例,串行會從上到下依次執行對應接口請求。

并行

通常,我們在需要保證代碼在多個異步處理之后執行,會用到:

Promise.all(promises:?[]).then(fun:?function);

Promise.all可以保證,promises數組中所有promise對象都達到resolve狀態,才執行then回調。

var?promises?=?function?()?{return?[1000,?2000,?3000].map(current?=>?{return?new?Promise(function?(resolve,?reject)?{setTimeout(()?=>?{console.log(current)},?current)})})
}Promise.all(promises()).then(()?=>?{console.log('end')
})

Promise.all 并發限制

這時候考慮一個場景:如果你的promises數組中每個對象都是http請求,而這樣的對象有幾十萬個。

那么會出現的情況是,你在瞬間發出幾十萬個http請求,這樣很有可能導致堆積了無數調用棧導致內存溢出。

這時候,我們就需要考慮對Promise.all做并發限制。

Promise.all并發限制指的是,每個時刻并發執行的promise數量是固定的,最終的執行結果還是保持與原來的Promise.all一致。

題目實現

思路分析

整體采用遞歸調用來實現:最初發送的請求數量上限為允許的最大值,并且這些請求中的每一個都應該在完成時繼續遞歸發送,通過傳入的索引來確定了urls里面具體是那個URL,保證最后輸出的順序不會亂,而是依次輸出。

代碼實現

function?multiRequest(urls?=?[],?maxNum)?{//?請求總數量const?len?=?urls.length;//?根據請求數量創建一個數組來保存請求的結果const?result?=?new?Array(len).fill(false);//?當前完成的數量let?count?=?0;return?new?Promise((resolve,?reject)?=>?{//?請求maxNum個while?(count?<?maxNum)?{next();}function?next()?{let?current?=?count++;//?處理邊界條件if?(current?>=?len)?{//?請求全部完成就將promise置為成功狀態,?然后將result作為promise值返回!result.includes(false)?&&?resolve(result);return;}const?url?=?urls[current];console.log(`開始?${current}`,?new?Date().toLocaleString());fetch(url).then((res)?=>?{//?保存請求結果result[current]?=?res;console.log(`完成?${current}`,?new?Date().toLocaleString());//?請求沒有全部完成,?就遞歸if?(current?<?len)?{next();}}).catch((err)?=>?{console.log(`結束?${current}`,?new?Date().toLocaleString());result[current]?=?err;//?請求沒有全部完成,?就遞歸if?(current?<?len)?{next();}});}});
}

推薦閱讀

我在阿里招前端,我該怎么幫你?(現在還可以加模擬面試群)
如何拿下阿里巴巴 P6 的前端 Offer
如何準備阿里P6/P7前端面試--項目經歷準備篇
大廠面試官常問的亮點,該如何做出?
如何從初級到專家(P4-P7)打破成長瓶頸和有效突破
若川知乎問答:2年前端經驗,做的項目沒什么技術含量,怎么辦?
若川知乎高贊:有哪些必看的 JS庫?

末尾

你好,我是若川,江湖人稱菜如若川,歷時一年只寫了一個學習源碼整體架構系列~(點擊藍字了解我)

  1. 關注若川視野,回復"pdf" 領取優質前端書籍pdf,回復"1",可加群長期交流學習

  2. 我的博客地址:https://lxchuan12.gitee.io?歡迎收藏

  3. 覺得文章不錯,可以點個在看呀^_^另外歡迎留言交流~

精選前端好文,伴你不斷成長

若川原創文章精選!可點擊

小提醒:若川視野公眾號面試、源碼等文章合集在菜單欄中間【源碼精選】按鈕,歡迎點擊閱讀,也可以星標我的公眾號,便于查找

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/276283.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/276283.shtml
英文地址,請注明出處:http://en.pswp.cn/news/276283.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

Jquery 中 ajaxSubmit使用講解

轉載鏈接&#xff1a;http://blog.csdn.net/h70614959/article/details/8810270 1 引入依賴腳本 <script type"text/javascript" src"/js/jquery/jquery.form.js"></script> //ajaxForm 依賴腳本<script type"text/javascript"…

接口與抽象類

接口中的屬性 默認是 public static final類型 就算你不加 默認也是存在 方法默認都是public abstract類型 不加默認也存在 比如 interface K { String abc"sdfd"; void add() throws Exception; } 實際上編譯時 是這樣 interface K { Public static final String a…

制造行業電子化簽約及印控一體化解決方案

當生產“智造化”、營銷“數字化”成為趨勢&#xff0c;生產制造型企業如何實現產品生產-銷售全流程提速降本&#xff1f;生產制造型企業為了在生產、營銷環節降低成本、提升效率&#xff0c;經營模式上常有如下特征&#xff1a;? 原料導向&#xff0c;多點協同生產&#xff1…

python 點擊按鈕采集圖片_python多線程采集圖片

cmd中運行>python untitled2.py 圖片的網站import requestsimport threadingfrom bs4 import BeautifulSoupimport sysimport osif len(sys.argv) ! 2:print("Usage : " )print(" python main.py [URL]" )exit(1)# config-starturl sys.argv[1]thre…

一起手寫Vue3核心模塊源碼,掌握閱讀源碼的正確方法

最近和一個獵頭聊天&#xff0c;說到現在前端供需脫節的境況。一方面用人方招不到想要的中高級前端&#xff0c;另一方面市場上有大量初級前端薪資要不上價。特別是用 Vue 框架的&#xff0c;因為好上手&#xff0c;所以很多人將 Vue 作為入門框架&#xff0c;但學得深的人并不…

jquery|javascript 回車事件

轉載鏈接&#xff1a;http://www.cnblogs.com/wenbo/archive/2011/08/20/2147014.html 轉載鏈接: http://www.2cto.com/kf/201202/119741.html jquery實現回車事件&#xff0c;代碼如下&#xff1a; 全局&#xff1a; $(function(){ document.onkeydown function(e){…

【Vegas原創】ASP 0131 不允許父路徑的解決

現象&#xff1a; Active Server Pages 錯誤 ASP 0131 不允許的父路徑 /admin/login.asp&#xff0c;行 2 包含文件 ../include/config.asp 不能用 .. 表示父目錄。 解決方案&#xff1a; IIS6>站點屬性>主目錄>配置>選項>啟用父路徑 轉載于:https://www.cnblog…

騰訊正式宣布成立技術委員會,要對組織架構下狠手

2019 年伊始&#xff0c;社交巨人騰訊正式宣布成立技術委員會&#xff0c;計劃在未來發力內部代碼的開源和協同&#xff0c;投入更多資金和精力在技術建設上。BAT 三家里一直被詬病技術建設落后的騰訊&#xff0c;終于開始補足自己的短板。這會對騰訊自身帶來什么&#xff1f;又…

2020 前端技術發展回顧

大家好&#xff0c;我是若川&#xff0c;今天給大家分享一篇來自阿里媽媽前端快爆的好文。另外&#xff0c;偷偷告訴你&#xff0c;截止今晚9點 點此抽獎送紅包還送3本比紅寶書還貴的書可以點擊文章末尾閱讀原文直達知乎鏈接&#xff0c;以下是正文2020 終究是一個不平凡的一年…

php驗證碼函數 使用imagestring() imagefttext()設置字體大小

轉載路徑&#xff1a;http://hi.baidu.com/asdasd_cn/item/62977d1e26ca36e85e53b142 生成驗證碼圖片的兩個函數 第一個是用imagestring bool imagestring ( resource $image , int $font , int $x , int $y , string $s , int $col ) imagestring() 用 col …

提取地圖中道路_非機器學習方法·從遙感影像中提取道路

本科低年級曾經做過一個提取道路線的題目。提供的數據&#xff1a;老師給我們了一幅學校周邊地區的影像&#xff0c;包括RGBInfrared四個波段。一開始使用的方法&#xff1a;當時尚青澀的我和小伙伴們一起使用KMEANS、閾值分割等等方法試圖將那道路從影像中分離出來。遇到的問題…

圖片尺寸自適應

圖片尺寸自適應 οnlοad"javascript:if(this.height>this.width){this.height100}else{this.width100} 轉載于:https://www.cnblogs.com/TangZhongxin/archive/2009/12/14/3942483.html

第一章知識點

第一章知識點 一&#xff1a;SQL語言 1&#xff0c;結構化查詢語言&#xff0c;是關系數據庫的標準語言 2&#xff0c;分類&#xff1a; 2.1&#xff1a;數據操作語言&#xff1a;DML&#xff1b;包括&#xff1a;即增刪查改&#xff1b;insert&#xff0c;update&#xff0c;d…

2021年,推薦這幾個優質公眾號碎片化學習

2021 年了&#xff0c;前端技術日新月異&#xff0c;發展迅速&#xff0c;前端公眾號是不是感覺越來越多了&#xff1f;在著辭舊迎新之際&#xff0c;這里盤點幾個前端開發工程師 2021 年必須關注的優質公眾號&#xff0c;希望對你有所幫助。大家可以像我一樣&#xff0c;利用碎…

php 處理表單里面的 單雙引號

轉載鏈接&#xff1a;http://developer.51cto.com/art/200911/165392.htm 我們今天要向大家介紹的是PHP magic_quotes_gpc的具體使用方法。大家都知道在PHP中一個特殊的函數魔術函數&#xff0c;它在引用的過程中只有在傳遞$_GET,$_POST,$_COOKIE時才會發生作用。 PHP函數preg…

阿里云插件新版發布,多特性助力提升開發者體驗

好消息&#xff01;阿里云 Cloud Toolkit 新版本于近日正式發布&#xff0c;推出了面向 IntelliJ 和 Eclipse 兩個平臺的新款插件&#xff0c;多個重大特性&#xff0c;持續提升開發者體驗&#xff0c;本文將帶大家快速預覽該新版本。 本文只挑選下面三個重大特性進行解讀&…

海鷗表表帶太長了怎么拆_表帶安裝,表帶太長了,怎么拆解和安裝?

表帶太長了&#xff0c;怎么拆解和安裝&#xff1f;1、準備拆表帶前請看清楚表帶背面的箭頭。2、有箭頭的帶粒是可拆卸的。3、擰松拆帶器把手&#xff0c;將表帶放入表槽&#xff0c;并保持拆帶器的鋼針對準針孔&#xff0c;如圖二&#xff0c;且拆帶器的鋼針順著箭頭的指示方向…

python類庫31[文件和目錄os+os.path+shutil]

一 常用函數 os模塊 os.sep 表示默認的文件路徑分隔符&#xff0c;windows為\, linux為/os.walk(spath): 用來遍歷目錄下的文件和子目錄os.listdir(dirname)&#xff1a;列出dirname下的目錄和文件os.mkdir() &#xff1a; 創建目錄os.makedirs(): 創建目錄&#xff0c;包含中間…

上一輪中獎信息公布

大家好&#xff0c;我是若川&#xff0c;抽獎活動其實挺耗時耗力的&#xff0c;尤其是留言和在看抽獎。比如我這篇文章寫完就已經23:50了。特別想說的是&#xff1a;有126人點擊了在看&#xff0c;但我的好友展示只有93人在看。說明還有30多人點擊了在看&#xff0c;但忘記加我…

css判斷不同分辨率顯示不同寬度布局實現自適應寬度

轉載鏈接&#xff1a;http://www.jb51.net/css/151312.html 點評&#xff1a;CSS判斷不同分辨率瀏覽器&#xff08;顯示屏幕&#xff09;顯示不同寬度布局CSS3技術支持IE6到IE8。將用到css3 media樣式進行判斷&#xff0c;但IE9以下版本不支持CSS3技術&#xff0c;這里DIVCSS5給…