記一道控制並行數的前端面試題

題目:請實現以下函數,能夠批量請求數據,全部的 url 地址在 urls 參數中,同時能夠經過 max 參數控制請求的並行數,當全部請求結束以後,須要執行 callback 回調函數,發送請求的函數能夠直接使用 fetch 便可html

function sendRequest(urls: strind[], max: number, callback: () => void) {

}
複製代碼

思路分析:數組

  1. 經過批量而且當全部請求結束後,在執行 callback 咱們初步肯定使用 Promise.all 能夠實現此功能,
  2. 因爲請求地址在 urls 數組中,由於是一個並行的問題,因此咱們能夠將請求地址進行按照 max 來進行分組,最後獲得多少組就執行多少次請求

具體代碼實現:bash

let bodyElement = document.body
    let urls = [
      'https://ss0.bdstatic.com/70cFuHSh_Q1YnxGkpoWK1HF6hhy/it/u=2580986389,1527418707&fm=27&gp=0.jpg',
      'https://ss3.bdstatic.com/70cFv8Sh_Q1YnxGkpoWK1HF6hhy/it/u=1995874357,4132437942&fm=27&gp=0.jpg',
      'https://ss0.bdstatic.com/70cFuHSh_Q1YnxGkpoWK1HF6hhy/it/u=2640393967,721831803&fm=27&gp=0.jpg',
      'https://ss1.bdstatic.com/70cFvXSh_Q1YnxGkpoWK1HF6hhy/it/u=1548525155,1032715394&fm=27&gp=0.jpg',
      'https://ss1.bdstatic.com/70cFvXSh_Q1YnxGkpoWK1HF6hhy/it/u=2434600655,2612296260&fm=27&gp=0.jpg',
      'https://ss1.bdstatic.com/70cFuXSh_Q1YnxGkpoWK1HF6hhy/it/u=2160840192,133594931&fm=27&gp=0.jpg'
    ]
    let max = 4
    let callback = () => {
      console.log('全部請求完成了')
    }
    // 定義一個分組函數 已 max 爲最大個數存儲在一個對象中
    let group = (urls, max) => {
      let groupObj = {}
      
      urls.forEach((item, index) => {
        let group = parseInt(index / max)
        if (groupObj[group]) {
          return groupObj[group].push(item)
        }
        groupObj[group] = [item]
      })
      
      return groupObj
    }
    
    function sendRequest(urls, max, callback) {
      let groupResult = group(urls, max)
      let currentIndex = 0
      // 使用 fetch 封裝請求
      let getFetch = source => {
        return source.map(item => fetch(item).then(res => res.blob()))
      }
      let send = () => {
        //  判斷有沒有當前組
        groupResult[currentIndex] &&
        Promise.all(getFetch(groupResult[currentIndex]))
          .then((body) => {
            callback()
            currentIndex++
            console.log(body, `第${currentIndex}次批量請求成功`)
            
            let html = ''
            
            body.forEach(item => {
              html += `<img src=${URL.createObjectURL(item)} />`
            })
            bodyElement.innerHTML += html
            
            // 用延時器是由於反應接口太快 以便於觀察
            setTimeout(() => {
              send()
            }, 1000)
          })
          .catch((err) => {
            console.log('error')
          })
      }

      send()
    }
    
    sendRequest(urls, max, callback)
複製代碼
功能已實現,你們有什麼好的實現方法能夠評論下,敬請指教。複製代碼
相關文章
相關標籤/搜索