理解Koa2中的async&await的用法


Posted in Javascript onFebruary 05, 2018

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

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
javascript-简单的计算器实现步骤分解(附图)
May 30 Javascript
js调试系列 控制台命令行API使用方法
Jun 18 Javascript
js实现延迟加载的方法
Jun 24 Javascript
基于JS2Image实现圣诞树代码
Dec 24 Javascript
浅谈js里面的InttoStr和StrtoInt
Jun 14 Javascript
JS实现根据文件字节数返回文件大小的方法
Aug 02 Javascript
AngularJS 单选框及多选框的双向动态绑定
Apr 20 Javascript
vue2.0 中#$emit,$on的使用详解
Jun 07 Javascript
javascript高级模块化require.js的具体使用方法
Oct 31 Javascript
原生JavaScript实现todolist功能
Mar 02 Javascript
Django+Vue实现WebSocket连接的示例代码
May 28 Javascript
vuex actions异步修改状态的实例详解
Nov 06 Javascript
zTree 树插件实现全国五级地区点击后加载的示例
Feb 05 #Javascript
使用vue如何构建一个自动建站项目
Feb 05 #Javascript
在 webpack 中使用 ECharts的实例详解
Feb 05 #Javascript
在Vue中使用echarts的方法
Feb 05 #Javascript
JavaScript中Object基础内部方法图
Feb 05 #Javascript
基于axios封装fetch方法及调用实例
Feb 05 #Javascript
JS设计模式之观察者模式实现实时改变页面中金额数的方法
Feb 05 #Javascript
You might like
php添加数据到xml文件的简单例子
2016/09/08 PHP
PHP使用mongoclient简单操作mongodb数据库示例
2019/02/08 PHP
thinkphp框架表单数组实现图片批量上传功能示例
2020/04/04 PHP
PHP开发api接口安全验证操作实例详解
2020/03/26 PHP
JS array 数组详解
2009/03/22 Javascript
Javascript学习笔记二 之 变量
2010/12/15 Javascript
纯js网页画板(Graphics)类简介及实现代码
2012/12/24 Javascript
浅析Js中的单引号与双引号问题
2013/11/06 Javascript
文字垂直滚动之javascript代码
2015/07/29 Javascript
Bootstrap入门书籍之(五)导航条、分页导航
2016/02/17 Javascript
js获取iframe中的window对象的实现方法
2016/05/20 Javascript
BOM系列第三篇之定时器应用(时钟、倒计时、秒表和闹钟)
2016/08/17 Javascript
JavaScript执行环境及作用域链实例分析
2018/08/01 Javascript
如何在 JavaScript 中更好地利用数组
2018/09/27 Javascript
微信小程序使用map组件实现解析经纬度功能示例
2019/01/22 Javascript
基于html+css+js实现简易计算器代码实例
2020/02/28 Javascript
[03:59]5分钟带你了解什么是DOTA2(第二期)
2017/02/07 DOTA
Python安装第三方库的3种方法
2015/06/21 Python
python中使用PIL制作并验证图片验证码
2018/03/15 Python
Python基于多线程实现ping扫描功能示例
2018/07/23 Python
python分数表示方式和写法
2019/06/26 Python
Python解决pip install时出现的Could not fetch URL问题
2019/08/01 Python
python各层级目录下import方法代码实例
2020/01/20 Python
基于canvas的骨骼动画的示例代码
2018/06/12 HTML / CSS
Html5 webview元素定位工具的实现
2020/08/07 HTML / CSS
Marriott中国:万豪国际酒店查询预订
2016/09/02 全球购物
英国领先的在线药房:Pharmacy First
2017/09/10 全球购物
Intersport西班牙:在线体育商店
2019/11/06 全球购物
英国最受欢迎的母婴精品品牌:JoJo Maman BéBé
2021/02/17 全球购物
《北大荒的秋天》教学反思
2014/04/14 职场文书
民主评议党员自我评议范文2014
2014/09/26 职场文书
广告设计专业毕业生自我鉴定
2014/09/27 职场文书
财务经理岗位职责范本
2015/04/08 职场文书
听证通知书
2015/04/24 职场文书
小学生大队委竞选稿
2015/11/20 职场文书
承诺书怎么写 ?
2019/04/16 职场文书