理解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 相关文章推荐
Jquery提交表单 Form.js官方插件介绍
Mar 01 Javascript
解析John Resig Simple JavaScript Inheritance代码
Dec 03 Javascript
谈一谈JS消息机制和事件机制的理解
Apr 14 Javascript
不间断循环滚动效果的实例代码(必看篇)
Oct 08 Javascript
Angular2库初探
Mar 01 Javascript
JavaScript数据结构之二叉树的删除算法示例
Apr 13 Javascript
jQuery extend()详解及简单实例
May 06 jQuery
Angularjs中数据绑定的实例详解
Aug 25 Javascript
jQuery实现checkbox的简单操作
Nov 18 jQuery
jQuery实现鼠标响应式透明度渐变动画效果示例
Feb 13 jQuery
Vue多系统切换实现方案
Jun 05 Javascript
webpack多入口多出口的实现方法
Aug 17 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弹出错误警告函数扩展性强
2014/01/17 PHP
ThinkPHP实现事务回滚示例代码
2014/06/23 PHP
Sublime里直接运行PHP配置方法
2014/11/28 PHP
PHP批量查询WordPress留言者E-mail地址实现方法
2015/02/15 PHP
TP5框架请求响应参数实例分析
2019/10/17 PHP
JavaScript中使用Object.prototype.toString判断是否为数组
2015/04/01 Javascript
深入学习JavaScript对象
2015/10/13 Javascript
移动端点击图片放大特效PhotoSwipe.js插件实现
2016/08/25 Javascript
BootstrapTable请求数据时设置超时(timeout)的方法
2017/01/22 Javascript
JSONP基础知识详解
2017/03/19 Javascript
Angular2学习教程之TemplateRef和ViewContainerRef详解
2017/05/25 Javascript
javascript checkbox/radio onchange不能兼容ie8处理办法
2017/06/13 Javascript
jQuery 实时保存页面动态添加的数据的示例
2017/08/14 jQuery
vue组件tabbar使用方法详解
2018/11/06 Javascript
一文快速详解前端框架 Vue 最强大的功能
2019/05/21 Javascript
解决vue中使用proxy配置不同端口和ip接口问题
2019/08/14 Javascript
通过vue写一个瀑布流插件代码实例
2019/09/07 Javascript
[01:20:37]FNATIC vs NIP 2019国际邀请赛小组赛 BO2 第一场 8.16
2019/08/19 DOTA
python在命令行下使用google翻译(带语音)
2014/01/16 Python
Python中函数的用法实例教程
2014/09/08 Python
利用Python实现图书超期提醒
2016/08/02 Python
Python网络编程使用select实现socket全双工异步通信功能示例
2018/04/09 Python
python分数表示方式和写法
2019/06/26 Python
用python求一个数组的和与平均值的实现方法
2019/06/29 Python
python输出电脑上所有的串口名的方法
2019/07/02 Python
python自动化UI工具发送QQ消息的实例
2019/08/27 Python
英国最大的在线蜡烛商店:Candles Direct
2019/03/26 全球购物
C++是不是类型安全的
2014/02/18 面试题
法院实习人员自我鉴定
2013/09/26 职场文书
2013届毕业生求职信范文
2013/11/20 职场文书
法人授权委托书格式
2014/04/08 职场文书
合作合同协议书范本
2015/01/27 职场文书
2015年售票员工作总结
2015/04/29 职场文书
创业计划书之青年旅馆
2019/09/23 职场文书
Ajax实现局部刷新的方法实例
2021/03/31 Javascript
斗罗大陆八大特殊魂兽,龙族始祖排榜首,第五最残忍(翠魔鸟)
2022/03/18 国漫