亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

如何在Koa2中使用async&await=

發布時間:2021-01-11 15:40:25 來源:億速云 閱讀:168 作者:Leah 欄目:web開發

今天就跟大家聊聊有關如何在Koa2中使用async&await=,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結了以下內容,希望大家根據這篇文章可以有所收獲。

Koa是一款非常著名的Node服務端框架,有1.x版本和2.x版本。前者使用了generator來進行異步操作,后者則用了最新的async/await方案

一開始使用這種寫法的時候,我遇到一個問題,代碼如下:

const Koa = require('koa');
const app = new Koa();

const doSomething = time => {
  return new Promise(resolve => {
    setTimeout(() => {
      resolve('task done!')
    }, time)
  })
}

// 用來打印請求信息
app.use((ctx, next) => {
  console.log(`${ctx.method}:::${ctx.url}`)
  next()
})

app.use(async ctx => {
  const result = await doSomething(3000)
  console.log(result);
  ctx.body = result
})

app.listen(3000);

讓我們測試一下:curl http://localhost:3000

期望結果:

(3秒后...)task done!

然而現實卻是:

(立即)
Not Found

什么鬼?為什么沒有按照預期執行?這就需要我們來理解下Koa中中間件是如何串聯起來的了。翻一下源碼,將middlewares串聯起來的代碼如下:

function compose (middleware) {
 return function (context, next) {
  // 這個index用來計數,防止next被多次調用
  let index = -1
  // 執行入口
  return dispatch(0)
  
  function dispatch (i) {
   // 如果next被多次調用,報異常
   if (i <= index) return Promise.reject(new Error('next() called multiple times'))
   index = i
   // 取出第一個middleware
   let fn = middleware[i]
   // 將最初傳入的next作為最后一個函數執行
   if (i === middleware.length) fn = next
   if (!fn) return Promise.resolve()
   try {
    /**
    這里就是關鍵了,Promise.resolve是什么意思呢?
     Promise.resolve方法有下面三種形式:
     
     Promise.resolve(value);
     Promise.resolve(promise);
     Promise.resolve(theanable);
     
    這三種形式都會產生一個新的Promise。其中:

    第一種形式提供了自定義Promise的值的能力,它與Promise.reject(reason)對應。兩者的不同,在于得到的Promise的狀態不同。

    第二種形式,提供了創建一個Promise的副本的能力。

    第三種形式,是將一個類似Promise的對象轉換成一個真正的Promise對象。它的一個重要作用是將一個其他實現的Promise對象封裝成一個當前實現的Promise對象。例如你正在用bluebird,但是現在有一個Q的Promise,那么你可以通過此方法把Q的Promise變成一個bluebird的Promise。第二種形式可以歸在第三種里面
    
    **/
    return Promise.resolve(fn(context, function next () {
     // 執行下一個middleware,返回結果也是一個Promise
     return dispatch(i + 1)
    }))
   } catch (err) {
    return Promise.reject(err)
   }
  }
 }
}

有了以上基礎,我們再來看一下之前的問題,為什么response沒有等到第二個middleware執行完成就立即返回了呢?

因為第一個middleware并不是一個異步函數啊。

由于每次next方法的執行,實際上都是返回了一個Promise對象,所以如果我們在某個middleware中執行了異步操作,要想等待其完成,就要在執行這個middleware之前添加await

那我們來改寫一下之前的代碼

app.use(async (ctx, next) => {
  console.log(`${ctx.method}:::${ctx.url}`)
  await next()
})

app.use(async ctx => {
  const result = await doSomething(3000)
  console.log(result);
  ctx.body = result
})

好了,沒有問題,一切如期望執行:clap:

錯誤處理

借助了Promise強大的功力,配合async/await語法,我們只需要把try/catch的操作寫在最外層的middleware中,就可以捕獲到之后所有中間件的異常!

app.use(async (ctx, next) => {
  try{
    await next()
  }catch(err){
    console.log(err)
  }
})

app.use(async (ctx)=>{
  throw new Error('something wrong!')
  ctx.body = 'Hello'
})

基于中間件鏈的完全控制,并且基于 Promise 的事實使得一切都變得容易操作起來。不再是到處的 if (err) return next(err) 而只有 promise

看完上述內容,你們對如何在Koa2中使用async&await=有進一步的了解嗎?如果還想了解更多知識或者相關內容,請關注億速云行業資訊頻道,感謝大家的支持。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

南漳县| 定西市| 博爱县| 江达县| 常德市| 西峡县| 大同县| 巴青县| 静安区| 长寿区| 津市市| 双柏县| 无为县| 浦县| 北川| 平湖市| 大理市| 天水市| 苏尼特左旗| 蓝田县| 会东县| 霍城县| 平塘县| 渝北区| 武城县| 诸暨市| 东乌珠穆沁旗| 内黄县| 尚义县| 广平县| 大关县| 鄄城县| 玛纳斯县| 浦县| 抚松县| 巴彦淖尔市| 天气| 灵宝市| 屏东县| 樟树市| 阿合奇县|