理解Koa2中的async&await的用法
Koa是一款非常著名的Node服务端框架,有1.x版本和2.x版本。前者使用了generator来进行异步操作,后者则用了最新的async/await方案
一开始使用这种写法的时候,我遇到一个问题,代码如下:
constKoa=require('koa'); constapp=newKoa(); constdoSomething=time=>{ returnnewPromise(resolve=>{ setTimeout(()=>{ resolve('taskdone!') },time) }) } //用来打印请求信息 app.use((ctx,next)=>{ console.log(`${ctx.method}:::${ctx.url}`) next() }) app.use(asyncctx=>{ constresult=awaitdoSomething(3000) console.log(result); ctx.body=result }) app.listen(3000);
让我们测试一下:curlhttp://localhost:3000
期望结果:
(3秒后...)taskdone!
然而现实却是:
(立即)
NotFound
什么鬼?为什么没有按照预期执行?这就需要我们来理解下Koa中中间件是如何串联起来的了。翻一下源码,将middlewares串联起来的代码如下:
functioncompose(middleware){ returnfunction(context,next){ //这个index用来计数,防止next被多次调用 letindex=-1 //执行入口 returndispatch(0) functiondispatch(i){ //如果next被多次调用,报异常 if(i<=index)returnPromise.reject(newError('next()calledmultipletimes')) index=i //取出第一个middleware letfn=middleware[i] //将最初传入的next作为最后一个函数执行 if(i===middleware.length)fn=next if(!fn)returnPromise.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。第二种形式可以归在第三种里面 **/ returnPromise.resolve(fn(context,functionnext(){ //执行下一个middleware,返回结果也是一个Promise returndispatch(i+1) })) }catch(err){ returnPromise.reject(err) } } } }
有了以上基础,我们再来看一下之前的问题,为什么response没有等到第二个middleware执行完成就立即返回了呢?
因为第一个middleware并不是一个异步函数啊。
由于每次next方法的执行,实际上都是返回了一个Promise对象,所以如果我们在某个middleware中执行了异步操作,要想等待其完成,就要在执行这个middleware之前添加await
那我们来改写一下之前的代码
app.use(async(ctx,next)=>{ console.log(`${ctx.method}:::${ctx.url}`) awaitnext() }) app.use(asyncctx=>{ constresult=awaitdoSomething(3000) console.log(result); ctx.body=result })
好了,没有问题,一切如期望执行:clap:
错误处理
借助了Promise强大的功力,配合async/await语法,我们只需要把try/catch的操作写在最外层的middleware中,就可以捕获到之后所有中间件的异常!
app.use(async(ctx,next)=>{ try{ awaitnext() }catch(err){ console.log(err) } }) app.use(async(ctx)=>{ thrownewError('somethingwrong!') ctx.body='Hello' })
基于中间件链的完全控制,并且基于Promise的事实使得一切都变得容易操作起来。不再是到处的if(err)returnnext(err)而只有promise
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持毛票票。
声明:本文内容来源于网络,版权归原作者所有,内容由互联网用户自发贡献自行上传,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任。如果您发现有涉嫌版权的内容,欢迎发送邮件至:czq8825#qq.com(发邮件时,请将#更换为@)进行举报,并提供相关证据,一经查实,本站将立刻删除涉嫌侵权内容。