async与await的用法详解

这次给大家带来async与await的用法详解,使用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

相信看了本文案例你已经掌握了方法,更多精彩请关注【创想鸟】其它相关文章!

推荐阅读:

datepicker怎么使用

mixin的高阶组件使用详解

ejsExcel模板在Vue.js中的使用

以上就是async与await的用法详解的详细内容,更多请关注【创想鸟】其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至253000106@qq.com举报,一经查实,本站将立刻删除。

发布者:PHP中文网,转转请注明出处:https://www.chuangxiangniao.com/p/2779599.html

(0)
上一篇 2025年3月8日 15:49:29
下一篇 2025年2月26日 01:34:45

AD推荐 黄金广告位招租... 更多推荐

相关推荐

  • js的作用域与预解析使用详解

    这次给大家带来js的作用域与预解析使用详解,js作用域与预解析使用的注意事项有哪些,下面就是实战案例,一起来看一下。 虽然,ES6在我们工作中应用得越来越广泛,但是还是很多项目保留着ES5的写法,所以,今天,带着大家重新巩固下ES5下的作用…

    编程技术 2025年3月8日
    200
  • js的闭包使用详解

    这次给大家带来js的闭包使用详解,js闭包使用的注意事项有哪些,下面就是实战案例,一起来看一下。 闭包 按中文的意思就是关上一个包的意思。如果我们把函数的作用域当做是一个包的话,那这个词很形象体现了它的作用 。函数的正常的执行流程是当函数中…

    编程技术 2025年3月8日
    200
  • Express与Koa2的使用详解

    这次给大家带来Express与Koa2的使用详解,Express与Koa2使用的注意事项有哪些,下面就是实战案例,一起来看一下。 知会上看到有个问题 《Express会被Koa2取代吗?》 。刚好对Express、koa有点小研究,于是简单…

    编程技术 2025年3月8日
    200
  • babel-loader文件预处理器使用详解

    loader官方解释是文件预处理器,通俗点说就是webpack在处理静态文件的时候,需要使用 loader 来加载各种文件,比如: html文件需要使用html-loader ,css 需要使用css-loader 、 style-load…

    编程技术 2025年3月8日
    200
  • JavaScript中五大常见函数详解

    在 javascript 中有一些问题会被拿出来经常讨论,这些问题每个人都有不同的思路,想要理解这些问题,最好的方法就是自己实现一遍,话不多说,开始正题。 数组扁平化 数组扁平化有很多方法,但最终最好的方法就是递归,实现一个指定深度的扁平化…

    编程技术 2025年3月8日
    200
  • JS原始值和引用值的储存方式详解

    本文主要和大家分享js原始值和引用值的储存方式详解,原始值指的是代表原始数据类型的值,也叫基本数据类型,引用值指的是复合数据类型的值。接下来通过示例代码给大家介绍js中原始值和引用值的储存方式,感兴趣的朋友一起看看吧。 在ECMAscrip…

    2025年3月8日
    200
  • JS的可折叠面板使用详解

    这次给大家带来js的可折叠面板使用详解,使用js可折叠面板的注意事项有哪些,下面就是实战案例,一起来看一下。 举个例子             Part1         I am part 1        Part2        I …

    编程技术 2025年3月8日
    200
  • JS的日期相关函数使用详解

    这次给大家带来js的日期相关函数使用详解,使用js日期相关函数的注意事项有哪些,下面就是实战案例,一起来看一下。 一、 日期相关函数 select current_date();select current_date;select curr…

    编程技术 2025年3月8日
    200
  • JS跨域处理详解

    本文主要和大家分享JS跨域处理详解,解决js的跨域问题可以使用Jsonp,希望能帮助到大家。 js不可以跨域请求数据。 什么是跨域: 1、域名不同 2、域名相同端口不同。 解决js的跨域问题可以使用Jsonp,使用js的特性绕过跨域请求。J…

    2025年3月8日
    200
  • JS中的闭包详解

    本文主要和大家分享js中的闭包详解,主要以代码的方式和大家讲解,希望能帮助到大家。 var n = 999;function f1() {  console.log(n);}f1() // 999 登录后复制 JavaScript有两种作用…

    编程技术 2025年3月8日
    200

发表回复

登录后才能评论