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

JS異步編程之generator與async/await語法糖詳解

 更新時(shí)間:2022年11月02日 09:29:17   作者:開水泡飯  
這篇文章主要為大家詳細(xì)介紹了JS異步編程中g(shù)enerator與async/await語法糖的使用,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以學(xué)習(xí)一下

Generator 異步方案

相比于傳統(tǒng)回調(diào)函數(shù)的方式處理異步調(diào)用,Promise最大的優(yōu)勢(shì)就是可以鏈?zhǔn)秸{(diào)用解決回調(diào)嵌套的問題。但是這樣寫依然會(huì)有大量的回調(diào)函數(shù),雖然他們之間沒有嵌套,但是還是沒有達(dá)到傳統(tǒng)同步代碼的可讀性。如果以下面的方式寫異步代碼,它是很簡(jiǎn)潔,也更容易閱讀的。

// like sync mode

try{
  const value1 = ajax('/api/url1')
  console.log(value1)
  const value2 = ajax('/api/url1')
  console.log(value2)
  const value3 = ajax('/api/url1')
  console.log(value3)
  const value4 = ajax('/api/url1')
  console.log(value4)
  const value5 = ajax('/api/url1')
  console.log(value5)
}catch(err){
  console.log(err)
}

在ES2015提供了生成器函數(shù)(Generator Function)它與普通函數(shù)的語法差別在于,在function語句之后和函數(shù)名之前,有一個(gè)“*”作為生成器函數(shù)的標(biāo)示符。

在我們?nèi)フ{(diào)用生成器函數(shù)的時(shí)候他并不會(huì)立即去執(zhí)行這個(gè)函數(shù),而是會(huì)得到一個(gè)生成器對(duì)象,直到我們手動(dòng)調(diào)用對(duì)象的next 方法,函數(shù)體才會(huì)開始執(zhí)行,我們可以使用關(guān)鍵字yield去向外返回一個(gè)值,我們可以在next方法的返回值中去拿到這個(gè)值。另外再返回的屬性中還有一個(gè)done關(guān)鍵字來表示生成器是否執(zhí)行完了,

yield不會(huì)像return一樣去結(jié)束函數(shù)的執(zhí)行,只是暫停函數(shù)的執(zhí)行,直到外接下一次調(diào)用next方法時(shí)才會(huì)繼續(xù)從yield位置往下執(zhí)行

function * foo () {
  console.log('start')
    yield 'foo'
}

???????const generator = foo()

const result = generator.next()

調(diào)用next方法的時(shí)候傳入了參數(shù)的話,所傳入的參數(shù)會(huì)作為yield關(guān)鍵字的返回值

function * foo () {
  console.log('start')
    // 我可以在這里接收next傳入的參數(shù)
    const res = yield 'foo'
  console.log(res) // 這是我傳入的參數(shù)
}

const generator = foo()

const result = generator.next('這是我傳入的參數(shù)')
console.log(result) // { value: 'foo', done: false }

如果我們調(diào)用了生成器函數(shù)的throw方法,這個(gè)方法會(huì)給生成器函數(shù)內(nèi)部拋出一個(gè)異常

function * foo () {
  console.log('start')
  // 我可以在這里接收next傳入的參數(shù)
  try {
    const res = yield 'foo'
    console.log(res) // 這是我傳入的參數(shù)
  } catch (err) {
    console.log(err.message) // 拋出錯(cuò)誤
  }
}

const generator = foo()

const result = generator.next('這是我傳入的參數(shù)')
console.log(result)

generator.throw(new Error('拋出錯(cuò)誤'))

利用生成器函數(shù)和Promise來實(shí)現(xiàn)異步編程的體驗(yàn)

function ajax(url) {
  return new Promise((resove, reject) => {
    var xhr = new XMLHttpRequest()
    xhr.open('GET', url)
    // 新方法可以直接接受一個(gè)j對(duì)象
    xhr.responseType = 'json'
    xhr.onload = function () {
      if (this.status === 200) {
        resove(this.response)
      } else {
        reject(new Error(this.statusText))
      }
    }
    xhr.send()
  })
}

function* main() {
  const user1 = yield ajax('/json1.json')
  console.log(user1)

  const user2 = yield ajax('/json2.json')
  console.log(user2)

  const user3 = yield ajax('/json3.json')
  console.log(user3)
}

const g = main()
const result = g.next()

result.value.then(data => {
  const result2 = g.next(data)

  if (result2.done) return
  result2.value.then(data2 => {
    const result3 = g.next(data2)

    if (result3.done) return
    result3.value.then(data3 => {
      g.next(data3)
    })
  })
})

很明顯生成器的執(zhí)行器可以使用遞歸的方式去調(diào)用

const g = main()

function handleResult(result) {
  if (result.done) return
  result.value.then(data => {
    handleResult(g.next(data))
  }, err => {
    g.throw(err)
  })
}

handleResult(g.next())

生成器函數(shù)的調(diào)用其實(shí)都是差不多的,所以我們可以寫一個(gè)比較通用的執(zhí)行器

function co(generator) {
  const g = generator()

  function handleResult(result) {
    if (result.done) return
    result.value.then(data => {
      handleResult(g.next(data))
    }, err => {
      g.throw(err)
    })
  }

  handleResult(g.next())
}


co(main)

當(dāng)然這樣的執(zhí)行器在社區(qū)中已經(jīng)有一個(gè)比較完善的庫了co。這種co的方案在2015年之前是特別流行的,后來在出了async/await語法糖之后,這種方案相對(duì)來講就沒有那么普及了。使用generator這種方法最明顯的變化就是異步調(diào)用回歸到扁平化了

async/await

有了generator之后js異步編程基本上與同步代碼有類似的體驗(yàn)了,但是使用generator這種異步方案還需要自己手動(dòng)去寫一個(gè)執(zhí)行器函數(shù),會(huì)比較麻煩。在ES2017的版本中新增了一個(gè)叫做async的函數(shù),它同樣提供了這種扁平化的編程體驗(yàn),并且是語言層面的標(biāo)準(zhǔn)的異步編程語法。其實(shí)async函數(shù)就是生成器函數(shù)更方便的語法糖,所以語法上給generator函數(shù)是類似的。

async function main() {
  try {
    const user1 = await ajax('/json1.json')
    console.log(user1)

    const user2 = await ajax('/json2.json')
    console.log(user2)

    const user3 = await ajax('/json3.json')
    console.log(user3)
  } catch (error) {
    console.log(error)
  }
}

main()

async 函數(shù)返回一個(gè)Promise對(duì)象,更利于對(duì)整體代碼控制

promise.then(() => {
  console.log('all completed')
}).catch(err => {
  console.log(err)
})

到此這篇關(guān)于JS異步編程之generator與async/await語法糖詳解的文章就介紹到這了,更多相關(guān)JS generator async/await內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • JavaScript實(shí)現(xiàn)淘寶購物件數(shù)選擇

    JavaScript實(shí)現(xiàn)淘寶購物件數(shù)選擇

    這篇文章主要為大家詳細(xì)介紹了JavaScript實(shí)現(xiàn)淘寶購物件數(shù)的選擇,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-08-08
  • js實(shí)現(xiàn)tab切換效果實(shí)例

    js實(shí)現(xiàn)tab切換效果實(shí)例

    這篇文章主要介紹了js實(shí)現(xiàn)的tab標(biāo)簽切換效果,功能非常簡(jiǎn)單,實(shí)現(xiàn)了點(diǎn)擊切換的效果,推薦給大家,有需要的小伙伴可以參考下。
    2015-09-09
  • Three.js獲取鼠標(biāo)點(diǎn)擊的三維坐標(biāo)示例代碼

    Three.js獲取鼠標(biāo)點(diǎn)擊的三維坐標(biāo)示例代碼

    本篇文章主要介紹了Three.js獲取鼠標(biāo)點(diǎn)擊的三維坐標(biāo)示例代碼。具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下。
    2017-03-03
  • element中的$confirm的使用

    element中的$confirm的使用

    這篇文章主要介紹了element中的$confirm的使用,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-04-04
  • 屏蔽相應(yīng)鍵盤按鈕操作

    屏蔽相應(yīng)鍵盤按鈕操作

    本篇文章主要是對(duì)屏蔽相應(yīng)鍵盤按鈕的操作代碼進(jìn)行了介紹,需要的朋友可以過來參考下,希望對(duì)大家有所幫助
    2014-03-03
  • Javascript UrlDecode函數(shù)代碼

    Javascript UrlDecode函數(shù)代碼

    將Url進(jìn)行編碼,前臺(tái)JS需要使用這段內(nèi)容,這時(shí)候就需要解碼了
    2010-01-01
  • Javascript基礎(chǔ)之?dāng)?shù)組的使用

    Javascript基礎(chǔ)之?dāng)?shù)組的使用

    這篇文章主要介紹了Javascript基礎(chǔ)之?dāng)?shù)組的使用的相關(guān)資料,介紹的非常詳解,具有參考借鑒價(jià)值,感興趣的朋友一起學(xué)習(xí)吧
    2016-05-05
  • Bootstarp風(fēng)格的toggle效果分享

    Bootstarp風(fēng)格的toggle效果分享

    這篇文章主要介紹了Bootstarp風(fēng)格的toggle效果分享的相關(guān)資料,需要的朋友可以參考下
    2016-02-02
  • 微信小程序?qū)崿F(xiàn)限制用戶轉(zhuǎn)發(fā)功能的實(shí)例代碼

    微信小程序?qū)崿F(xiàn)限制用戶轉(zhuǎn)發(fā)功能的實(shí)例代碼

    這篇文章主要介紹了微信小程序?qū)崿F(xiàn)限制用戶轉(zhuǎn)發(fā)的實(shí)例代碼,通過截圖加實(shí)例代碼給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-02-02
  • 根據(jù)配置文件加載js依賴模塊

    根據(jù)配置文件加載js依賴模塊

    這篇文章主要介紹了根據(jù)配置文件加載js依賴模塊,解決方法是筆者自己的思路,拋磚引玉,需要的朋友可以參考下
    2014-12-12

最新評(píng)論