欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

JavaScript異步編程常見(jiàn)面試題匯總

 更新時(shí)間:2023年02月08日 10:03:34   作者:腹黑的可樂(lè)  
本文將帶大家學(xué)習(xí)異步編程這一塊內(nèi)容,鑒于異步編程是js中至關(guān)重要的內(nèi)容,所以我們將學(xué)習(xí)異步編程涉及到的重點(diǎn)和難點(diǎn),同時(shí)這一塊內(nèi)容也是面試常考范圍,希望對(duì)大家有所幫助

在上一節(jié)中我們了解了常見(jiàn)的es6語(yǔ)法的一些知識(shí)點(diǎn)。這一章節(jié)我們將會(huì)學(xué)習(xí)異步編程這一塊內(nèi)容,鑒于異步編程是js中至關(guān)重要的內(nèi)容,所以我們將會(huì)用三個(gè)章節(jié)來(lái)學(xué)習(xí)異步編程涉及到的重點(diǎn)和難點(diǎn),同時(shí)這一塊內(nèi)容也是面試??挤秶?/p>

并發(fā)(concurrency)和并行(parallelism)的區(qū)別

面試題 并發(fā)和并行的區(qū)別?

異步和這一小節(jié)的知識(shí)點(diǎn)其實(shí)并不是一個(gè)概念,但是這個(gè)兩個(gè)名詞確實(shí)是很多人混淆的知識(shí)點(diǎn),其實(shí)混淆的原因可能只是兩個(gè)名詞在中文的相似,在英文上來(lái)說(shuō)完全是不同的單詞。

并發(fā)是宏觀概念,我分別有任務(wù)A和任務(wù)B,在一段時(shí)間內(nèi)通過(guò)任務(wù)間的切換完成了這兩個(gè)任務(wù),這種情況就可以成為并發(fā)。

并行是微觀概念,假設(shè)cpu中存在兩個(gè)核心,那么我就可以同時(shí)完成任務(wù)A,B。同時(shí)完成多個(gè)任務(wù)的情況就可以稱之為并行。

回調(diào)函數(shù)(callback)

面試題: 什么是回調(diào)函數(shù)?回調(diào)函數(shù)有什么缺點(diǎn)?如何解決回調(diào)地獄問(wèn)題?

回調(diào)函數(shù)應(yīng)該是大家經(jīng)常使用到的,以下代碼是回調(diào)函數(shù)的例子:

ajax(url,()=>{
    //處理邏輯
})

但是回調(diào)函數(shù)有個(gè)致命的弱點(diǎn),就是容易寫出回調(diào)地獄,假設(shè)多個(gè)請(qǐng)求存在依賴性,你可能就會(huì)寫出如下代碼:

ajax(url,()=>{
    ajax(url,()=>{})
})

以上代碼看起來(lái)不利于閱讀和維護(hù),當(dāng)然你可能會(huì)說(shuō)解決這個(gè)問(wèn)題還不簡(jiǎn)單,把函數(shù)分開(kāi)來(lái)寫不就得了

function firstAjax(){
    ajax(url1,()=>{
        secondAjax()
    })

}
function second(){
    ajax(url2,()=>{

    })
}
ajax(url,()=>{
    firstAjax()
})

以上代碼看上去有利于閱讀了,但是還是沒(méi)有解決根本問(wèn)題

回調(diào)地獄得根本問(wèn)題是:

  • 嵌套函數(shù)存在耦合性,一旦有改動(dòng),就會(huì)牽一發(fā)而動(dòng)全身
  • 嵌套函數(shù)一多就很難處理錯(cuò)誤

當(dāng)然,回調(diào)函數(shù)還存在著別的缺點(diǎn),比如不能使用try catch捕獲錯(cuò)誤,不能直接return。

Generator

面試題:你理解的generator是什么?

Generator算是es6中難理解的概念之一了,Generator最大的特點(diǎn)就是可以控制函數(shù)的執(zhí)行。在這一小節(jié)中我們不會(huì)講什么是Generator,而把重點(diǎn)放在Generator的一些容易困惑的地方。

function  *foo(){
    let y = 2*(yield(x+1))
    let z = yield(y/3)
    return (x+y+z)
}
let it = foo(5)
console.log(it.next())
console.log(it.next(12))
console.log(it.next(13))

你也許會(huì)疑惑為什么會(huì)產(chǎn)生與你預(yù)想不同的值,接下來(lái)就讓我為你逐行代碼分析原因

  • 首先 Generator 函數(shù)調(diào)用和普通函數(shù)不同,它會(huì)返回一個(gè)迭代器
  • 當(dāng)執(zhí)行第一次 next 時(shí),傳參會(huì)被忽略,并且函數(shù)暫停在 yield (x + 1) 處,所以返回 5 + 1 = 6
  • 當(dāng)執(zhí)行第二次 next 時(shí),傳入的參數(shù)等于上一個(gè) yield 的返回值,如果你不傳參,yield 永遠(yuǎn)返回 undefined。此時(shí) let y = 2 12,所以第二個(gè) yield 等于 2 12 / 3 = 8
  • 當(dāng)執(zhí)行第三次 next 時(shí),傳入的參數(shù)會(huì)傳遞給 z,所以 z = 13, x = 5, y = 24,相加等于 42

Generator 函數(shù)一般見(jiàn)到的不多,其實(shí)也于他有點(diǎn)繞有關(guān)系,并且一般會(huì)配合 co 庫(kù)去使用。當(dāng)然,我們可以通過(guò) Generator 函數(shù)解決回調(diào)地獄的問(wèn)題,可以把之前的回調(diào)地獄例子改寫為如下代碼:

function *fetch() {
    yield ajax(url, () => {})
    yield ajax(url1, () => {})
    yield ajax(url2, () => {})
}
let it = fetch()
let result1 = it.next()
let result2 = it.next()
let result3 = it.next()

Promise

翻譯過(guò)來(lái)就是承諾的意思,這個(gè)承諾會(huì)在未來(lái)有一個(gè)確切的答復(fù),并且該承諾有三種狀態(tài),分別是:

  • 等待中(pending)
  • 完成了 (resolved)
  • 拒絕了(rejected)

這個(gè)承諾一旦從等待狀態(tài)變成其他狀態(tài)就永遠(yuǎn)不能更改狀態(tài)了,也就是說(shuō)一旦狀態(tài)編為resolved后就不能再次改變

new Promise((resolve, reject) => {
  resolve('success')
  // 無(wú)效
  reject('reject')
})

當(dāng)我們?cè)跇?gòu)造 Promise 的時(shí)候,構(gòu)造函數(shù)內(nèi)部的代碼是立即執(zhí)行的

new Promise((resolve, reject) => {
  console.log('new Promise')
  resolve('success')
})
console.log('finifsh')
// new Promise -> finifsh

Promise 實(shí)現(xiàn)了鏈?zhǔn)秸{(diào)用,也就是說(shuō)每次調(diào)用 then 之后返回的都是一個(gè) Promise,并且是一個(gè)全新的 Promise,原因也是因?yàn)闋顟B(tài)不可變。如果你在 then 中 使用了 return,那么 return 的值會(huì)被 Promise.resolve() 包裝

Promise.resolve(1)
  .then(res => {
    console.log(res) // => 1
    return 2 // 包裝成 Promise.resolve(2)
  })
  .then(res => {
    console.log(res) // => 2
  })

當(dāng)然了,Promise 也很好地解決了回調(diào)地獄的問(wèn)題,可以把之前的回調(diào)地獄例子改寫為如下代碼:

ajax(url)
  .then(res => {
      console.log(res)
      return ajax(url1)
  }).then(res => {
      console.log(res)
      return ajax(url2)
  }).then(res => console.log(res))

前面都是在講述 Promise 的一些優(yōu)點(diǎn)和特點(diǎn),其實(shí)它也是存在一些缺點(diǎn)的,比如無(wú)法取消 Promise,錯(cuò)誤需要通過(guò)回調(diào)函數(shù)捕獲。

async 及 await

面試題:async 及 await 的特點(diǎn),它們的優(yōu)點(diǎn)和缺點(diǎn)分別是什么?await 原理是什么?

一個(gè)函數(shù)如果加上 async ,那么該函數(shù)就會(huì)返回一個(gè) Promise

async function test() {
  return "1"
}
console.log(test()) // -> Promise {<resolved>: "1"}

async 就是將函數(shù)返回值使用 Promise.resolve() 包裹了下,和 then 中處理返回值一樣,并且 await 只能配套 async 使用

async function test() {
  let value = await sleep()
}

async 和 await 可以說(shuō)是異步終極解決方案了,相比直接使用 Promise 來(lái)說(shuō),優(yōu)勢(shì)在于處理 then 的調(diào)用鏈,能夠更清晰準(zhǔn)確的寫出代碼,畢竟寫一大堆 then 也很惡心,并且也能優(yōu)雅地解決回調(diào)地獄問(wèn)題。當(dāng)然也存在一些缺點(diǎn),因?yàn)?await 將異步代碼改造成了同步代碼,如果多個(gè)異步代碼沒(méi)有依賴性卻使用了 await 會(huì)導(dǎo)致性能上的降低。

async function test() {
  // 以下代碼沒(méi)有依賴性的話,完全可以使用 Promise.all 的方式
  // 如果有依賴性的話,其實(shí)就是解決回調(diào)地獄的例子了
  await fetch(url)
  await fetch(url1)
  await fetch(url2)
}

下面來(lái)看一個(gè)使用 await 的例子:

let a = 0
let b = async () => {
  a = a + await 10
  console.log('2', a) // -> '2' 10
}
b()
a++
console.log('1', a) // -> '1' 1

對(duì)于以上代碼你可能會(huì)有疑惑,讓我來(lái)解釋下原因

  • 首先b先執(zhí)行,在執(zhí)行await 10之前變量a還是0,因?yàn)閍wait內(nèi)部實(shí)現(xiàn)了generator,generator會(huì)保留堆棧中東西,所以這個(gè)時(shí)候a = 0被保存下來(lái)
  • 因?yàn)閍wait是異步操作,后來(lái)的表達(dá)式不返回promise的話,就會(huì)包裝成Promise.resolve(返回值),然后去執(zhí)行函數(shù)外的同步代碼
  • 同步代碼執(zhí)行完畢后開(kāi)始執(zhí)行異步代碼,將保存下來(lái)的值拿出來(lái)使用,這時(shí)候 a = 0 + 10

上述解釋中提到了 await 內(nèi)部實(shí)現(xiàn)了 generator,其實(shí) await 就是 generator 加上 Promise 的語(yǔ)法糖,且內(nèi)部實(shí)現(xiàn)了自動(dòng)執(zhí)行 generator。如果你熟悉 co 的話,其實(shí)自己就可以實(shí)現(xiàn)這樣的語(yǔ)法糖。

常用定時(shí)器

面試題: setTimeout,setInterval,requestAnimationFrame 各有什么特點(diǎn)?

異步編程當(dāng)然少不了定時(shí)器,常見(jiàn)的定時(shí)器函數(shù)有setTimeout,setInterval,requestAnimationFrame。我們先來(lái)講講最常用的setTimeout,很多人認(rèn)為setTimeout是延遲多久,那就應(yīng)該是多久后執(zhí)行。

其實(shí)這個(gè)觀點(diǎn)是錯(cuò)誤的,因?yàn)閖s是單線程執(zhí)行的,如果前面的代碼影響了性能,就會(huì)導(dǎo)致setTimeout不會(huì)按期執(zhí)行。當(dāng)然了,我們可以通過(guò)代碼修正setTimeout,從而使定時(shí)器相對(duì)準(zhǔn)確

let period = 60 * 1000 * 60 * 2
let startTime = new Date().getTime()
let count = 0
let end = new Date().getTime() + period
let interval = 1000
let currentInterval = interval

function loop() {
  count++
  // 代碼執(zhí)行所消耗的時(shí)間
  let offset = new Date().getTime() - (startTime + count * interval);
  let diff = end - new Date().getTime()
  let h = Math.floor(diff / (60 * 1000 * 60))
  let hdiff = diff % (60 * 1000 * 60)
  let m = Math.floor(hdiff / (60 * 1000))
  let mdiff = hdiff % (60 * 1000)
  let s = mdiff / (1000)
  let sCeil = Math.ceil(s)
  let sFloor = Math.floor(s)
  // 得到下一次循環(huán)所消耗的時(shí)間
  currentInterval = interval - offset 
  console.log('時(shí):'+h, '分:'+m, '毫秒:'+s, '秒向上取整:'+sCeil, '代碼執(zhí)行時(shí)間:'+offset, '下次循環(huán)間隔'+currentInterval) 

  setTimeout(loop, currentInterval)
}

setTimeout(loop, currentInterval)

接下來(lái)我們來(lái)看 setInterval,其實(shí)這個(gè)函數(shù)作用和 setTimeout 基本一致,只是該函數(shù)是每隔一段時(shí)間執(zhí)行一次回調(diào)函數(shù)。

通常來(lái)說(shuō)不建議使用 setInterval。第一,它和 setTimeout 一樣,不能保證在預(yù)期的時(shí)間執(zhí)行任務(wù)。第二,它存在執(zhí)行累積的問(wèn)題,請(qǐng)看以下偽代碼

function demo() {
  setInterval(function(){
    console.log(2)
  },1000)
  sleep(2000)
}
demo()

以上代碼在瀏覽器環(huán)境中,如果定時(shí)器執(zhí)行過(guò)程中出現(xiàn)了耗時(shí)操作,多個(gè)回調(diào)函數(shù)會(huì)在耗時(shí)操作結(jié)束以后同時(shí)執(zhí)行,這樣可能就會(huì)帶來(lái)性能上的問(wèn)題。

如果你有循環(huán)定時(shí)器的需求,其實(shí)完全可以通過(guò) requestAnimationFrame 來(lái)實(shí)現(xiàn)

function setInterval(callback, interval) {
  let timer
  const now = Date.now
  let startTime = now()
  let endTime = startTime
  const loop = () => {
    timer = window.requestAnimationFrame(loop)
    endTime = now()
    if (endTime - startTime >= interval) {
      startTime = endTime = now()
      callback(timer)
    }
  }
  timer = window.requestAnimationFrame(loop)
  return timer
}

let a = 0
setInterval(timer => {
  console.log(1)
  a++
  if (a === 3) cancelAnimationFrame(timer)
}, 1000)

首先 requestAnimationFrame 自帶函數(shù)節(jié)流功能,基本可以保證在 16.6 毫秒內(nèi)只執(zhí)行一次(不掉幀的情況下),并且該函數(shù)的延時(shí)效果是精確的,沒(méi)有其他定時(shí)器時(shí)間不準(zhǔn)的問(wèn)題,當(dāng)然你也可以通過(guò)該函數(shù)來(lái)實(shí)現(xiàn) setTimeout。

以上就是JavaScript異步編程常見(jiàn)面試題匯總的詳細(xì)內(nèi)容,更多關(guān)于JavaScript異步編程的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • 深入理解JavaScript系列(27):設(shè)計(jì)模式之建造者模式詳解

    深入理解JavaScript系列(27):設(shè)計(jì)模式之建造者模式詳解

    這篇文章主要介紹了深入理解JavaScript系列(27):設(shè)計(jì)模式之建造者模式詳解,建造者模式可以將一個(gè)復(fù)雜對(duì)象的構(gòu)建與其表示相分離,使得同樣的構(gòu)建過(guò)程可以創(chuàng)建不同的表示,需要的朋友可以參考下
    2015-03-03
  • JS中改變this指向的方法(call和apply、bind)

    JS中改變this指向的方法(call和apply、bind)

    this是javascript的一個(gè)關(guān)鍵字,隨著函數(shù)使用場(chǎng)合不同,this的值會(huì)發(fā)生變化。但是總有一個(gè)原則,那就是this指的是調(diào)用函數(shù)的那個(gè)對(duì)象,通過(guò)本文給大家介紹JS中改變this指向的方法(call和apply、bind),需要的朋友參考下
    2016-03-03
  • js獲取 type=radio 值的方法

    js獲取 type=radio 值的方法

    這篇文章主要介紹了js如何獲取type=radio值,需要的朋友可以參考下
    2014-05-05
  • JavaScript變量聲明詳解

    JavaScript變量聲明詳解

    本文詳細(xì)向大家介紹了javascript變量聲明,并通過(guò)示例進(jìn)行了具體分析,是篇非常不錯(cuò)的文章,這里推薦給剛?cè)腴T的jser。
    2014-11-11
  • javascript vvorld 在線加密破解方法

    javascript vvorld 在線加密破解方法

    朋友公司開(kāi)發(fā)的在線JS加密站點(diǎn),內(nèi)測(cè)中,自己試過(guò)不能找到加密后的源代碼,不知道還有那位大大能夠破解
    2008-11-11
  • js定時(shí)調(diào)用方法成功后并停止調(diào)用示例

    js定時(shí)調(diào)用方法成功后并停止調(diào)用示例

    這篇文章主要介紹了js定時(shí)調(diào)用方法成功后并停止調(diào)用的實(shí)現(xiàn),需要的朋友可以參考下
    2014-04-04
  • D3.js 從P元素的創(chuàng)建開(kāi)始(顯示可加載數(shù)據(jù))

    D3.js 從P元素的創(chuàng)建開(kāi)始(顯示可加載數(shù)據(jù))

    D3是一個(gè)基于數(shù)據(jù)操作的可視化js庫(kù),認(rèn)識(shí)d3,就從最基礎(chǔ)的顯示可加載數(shù)據(jù)談起,需要的朋友可以參考下
    2014-10-10
  • js封裝tab標(biāo)簽頁(yè)實(shí)例分享

    js封裝tab標(biāo)簽頁(yè)實(shí)例分享

    本篇文章主要分享了js封裝tab標(biāo)簽頁(yè)的示例代碼,具有很好的參考價(jià)值,需要的朋友一起來(lái)看下吧
    2016-12-12
  • 使用element-plus時(shí)重寫樣式不起作用的問(wèn)題及解決方法

    使用element-plus時(shí)重寫樣式不起作用的問(wèn)題及解決方法

    這篇文章給大家介紹使用element-plus時(shí)重寫樣式不起作用的問(wèn)題及解決方法,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧
    2023-09-09
  • javascript中數(shù)組的常用算法深入分析

    javascript中數(shù)組的常用算法深入分析

    這篇文章主要給大家介紹了關(guān)于javascript中數(shù)組的常用算法的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者使用javascript具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2019-03-03

最新評(píng)論