JavaScript Async/Await: 串行,并行和复杂流

JavaScript Async/Await: 串行,并行和复杂流

如果你有ASP.NET MVC编程语言的从业经历,你应该很熟悉C#的关键字async/await。现在JavaScript 也有同样的特性了。在async/await关键字还没有出现在JavaScript时,我们通常使用回调函数和Promise来处理异步代码。async/await构建与Promise对象和非阻塞特性之上。它的强大之处在于提供了一种语法使其写异步代码如同同步代码一般体验。在这篇文章中,我们将遍历串行/链式和并行流,并使用async/await实现一个复杂的流。

async/await 结构

考虑下面返回Promise对象的函数

function myPromise() {
  return new Promise(function(resolve, reject) {
    // do your async Job here
    if (/* all is well */) {
      resolve('done')
    } else {
      reject(Error('Failed!'))
    }
  })
}

使用async/await

async function myAsyncFunc() {
  try {
    let result = await myPromise()
    console.log(result)
  } catch (err) {
    console.log(err)
  }
}

myAsyncFunc()

解释:

  • async关键字在函数之前使用,这意味着该函数必然会返回一个Promise对象,如果函数返回的不是Promise对象,则会被包装为一个Promise对象。
  • 关键字await在调用Promise函数的前面使用,它让JavaScript等待,直到Promise对象的状态变为onFulfilled并返回它的结果。
  • 通常使用try...catch来处理错误,在Promise对象被拒绝的案例中,代码执行将直接跳转到catch代码块。
  • await关键字只能在async函数内部使用,而不能在代码的顶层上下文使用,因为顶层上下文没有在async函数内部。

考虑下面的Promise函数:

function doJob(x, sec) {
  return new Promise(resolve => {
    console.log('Start:', x)
    setTimeout(() => {
      console.log('End', x)
      resolve(x)
    }, sec * 1000);
  })
}

我们将x和代表时间的参数sec传递到函数中,它将在给定时间之后返回x

串行、链式

如果你想要按顺序执行不同的Promise,或者说,一个接着一个执行。我们可以使用async/await语法像写同步代码一样很容易的写出来。

async function SerialFlow() {
  let result1 = await doJob(1, 1)
  let result2 = await doJob(2, 2)
  let result3 = await doJob(3, 3)

  let finalResult = result1 + result2 + result3

  console.log(finalResult)

  return finalResult
}

SerialFlow()

在上面的函数中,首先等待了1秒后1被执行,然后等待了2秒后开始执行2,最后执行3。1,2,3是按顺序执行的。因此总时间是6秒。

控制台输入如下:

Start: 1
End: 1
Start: 2
End: 2
Start: 3
End: 3
6

你可以使用循环来做到同样的目的:

async function SerialLoopFlow (jobs) {
  let finalResult = 0

  for (const job of jobs) {
    let result = await doJob(job, job)
    finalResult += result
  }
  console.log(finalResult)
}

SerialLoopFlow([1, 2, 3])

输出结果和上一个函数四同样的

你也可以使用Array.prototype.reduce来达到同样的目的:

async function SerialReduceFlow(jobs) {
  let finalResult = await jobs.reduce(async (total, job) => {
    return await total + await doJob(job, job)
  }, 0)

  console.log(finalResult)
}

SerialReduceFlow([1, 2, 3])

并行、并发流

如果你想要使用并行的方式执行同样的代码:

async function ParallelFlow() {
  let result1 = doJob(1, 1)
  let result2 = doJob(2, 2)
  let result3 = doJob(3, 3)

  let finalResult = await result1 + await result2 + await result3

  console.log(finalResult)

  return finalResult
}

ParallelFlow()

同时开始,在1秒后1执行结束,2,3分别在2秒和3秒结束。 总时间是3秒。

控制台输出如下:

Start: 1
Start: 2
Start: 3
End: 1
End: 2
End: 3
6

如果你认为是await处于同一行才导致的并行执行。那么,你想错了。试考虑如下代码:

async function ParallelFlow() {
  let result1 = doJob(1, 1)
  let result2 = doJob(2, 2)
  let result3 = doJob(3, 3)

  let r1 = await result1
  let r2 = await result2
  let r3 = await result3

  let finalResult = r1 + r2 + r3

  console.log(finalResult)

  return finalResult
}

ParallelFlow()

上面的程序是相同的,并行执行所有三个程序。因此关键点是,如果你使用了await doJob(1, 1) 因此它是按顺序执行的。如果你将它分配到一个变量上,并在之后await这个变量,则执行便是并行的。

对于数组,你可以使用Array.prototype.map并行的运行它。

async function ParallelMapFlow(jobs) {
  let results = jobs.map(async (job) => await doJob(job, job))
  let finalResult = 0
  for (const result of results) {
    finalResult += (await result)
  }

  console.log(finalResult)
}

ParallelMapFlow([1, 2, 3])

输出是一样的,3秒内函数将执行完成。你也可以使用Promise.allawait达到同样的效果

async function ParallelPromiseAllFlow(jobs) {
  let promises = jobs.map((job) => doJob(job, job))
  let result = await Promise.all(promises)
  let finalResult = 0

  for (const result of results) {
    finalResult += (await result)
  }

  console.log(finalResult)
}

复杂流

在实际的实现中,要求不是直截了当的。它是串行和并行流的组合。 让我们来看看下面的流程:
complex flow

1,2是串行的,然后3,4,5和6,7的串行是并行的。让我们使用async/await来实现它吧。

1,2串行:

let result1 = await doJob(1, 1)
let result2 = await doJob(2, 1)

6,7串行:

let Flow6_7 = async function() {
  let result6 = await doJob(6, 1)
  let result7 = await doJob(7, 1)

  return result6 + result7
}

现在3,4,5和6_7在并行执行:

let promises = [doJob(3,2), doJob(4, 2), doJob(5, 2), Flow6_7()]
let results = promises.map(async (job) => await job)

因为6和7都是1秒,所以3、4、5需要2秒的持续时间,这样所有的都可以在同一时间结束。

我们对所有的方法做求和操作

let finalResult = result1 + result2
for (const result of results) {
  finalResult += (await result)
}
console.log(finalResult)

使用try...catch处理错误,下面是完整的代码:

async function complexFlow() {
  try {
 
    let result1 = await doJob(1, 1)
    let result2 = await doJob(2, 1)
 
    let Flow6_7 = async function() {
      let result6 = await doJob(6, 1)
      let result7 = await doJob(7, 1)
      return result6 + result7
    }
 
    let promises = [doJob(3, 2), doJob(4, 2), doJob(5, 2), Flow6_7()];
    let results = promises.map(async (job) => await job)
    let finalResult = result1 + result2
    for (const result of results) {
      finalResult += (await result)
    }
    console.log(finalResult)
    return finalResult
 
  } catch (err) {
    console.log(err)
  }
}
 
complexFlow()

这是控制台的输出:
在这里插入图片描述
为了验证错误捕获,我们在特定的情况下抛出一个错误试试:

function doJob(x, sec) {
  if (x == 6) {
    throw Error('test error')
  } 
  return new Promise(resolve => {
    console.log('Start: ' + x)
    setTimeout(() => {
      console.log('End: ' + x)
      resolve(x)
    }, sec * 1000)
  })
}

然后控制台将会输出如下信息:

Error: test error
    at doJob ((index):69)
    at Flow6_7 ((index):87)
    at complexFlow ((index):92)

结论

Async/await确实简化了处理Promises的工作,并通过相同的try...catch处理同步和异步错误,并显示了完整的错误堆栈,这对调试错误有很大帮助。Node 8 LTS已经支持了Async/await的全部特性。你有在使用吗?你愿意分享你的经验吗?

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: JavaScriptasync/await 是用来处理异步操作的一种语法糖。它允许我们在异步函数中使用同步的写法。 我们可以使用 async 关键字来声明一个 async 函数,在 async 函数中,我们可以使用 await 关键字来等待一个异步操作的结果。 例如,我们可以使用 async/await 来等待一个 HTTP 请求的响应: ``` async function getData() { const response = await fetch('https://example.com/data.json'); const data = await response.json(); return data; } ``` 使用 async/await 可以让我们使用同步的写法来处理异步操作,这样就可以避免使用回调函数和 Promise.then() 方法了。 注意:async/await 只能在函数中使用,不能在全局作用域中使用。 ### 回答2: JavaScriptasync/await是一种处理异步操作的新特性。它是在ECMAScript 2017(ES8)中引入的,用于简化Promise的使用和处理异步代码的方式。 async/await的使用方式很简单,只需要在一个函数前面加上async关键字,这样就表明该函数是一个异步函数。在异步函数中,我们可以使用await关键字来等待一个Promise对象的完成,并且可以将异步操作的结果直接赋值给一个变量。 使用async/await可以使异步代码看起来更像同步代码,这样可以提高代码的可读性和易维护性。其实现原理是通过将异步操作放在一个微任务队列中,然后在等待异步操作完成的过程中,可以继续执行后续的代码。 在使用async/await时,我们可以使用try-catch语句来捕获异步操作中产生的异常。这使得可以更方便地处理和处理异步操作中的错误。 另外,async/await还支持并行执行多个异步操作。通过使用Promise.all()方法,我们可以将多个Promise对象包装成一个新的Promise对象,然后使用await关键字等待新的Promise对象的完成。 需要注意的是,async/await只能改变异步操作的执行顺序,而不能改变异步操作本身的执行时间。所以在使用时,还是需要注意避免长时间的阻塞操作,以免影响程序的性能和响应速度。 总结来说,JavaScriptasync/await是一种用于简化处理异步操作的语法糖。它使得异步代码更容易阅读和写入,并提供了更方便的错误处理和并行执行多个异步操作的能力。这是一个非常有用的特性,在现代的JavaScript中被广泛使用。 ### 回答3: JavaScript中的async/await是一种用于处理异步操作的编程模式。它是ES8(ECMAScript 2017)中引入的新特性,旨在简化异步编程,使代码更易于理解和维护。 async/await是建立在Promise的基础上的。通过在函数前面加上async关键字,可以将函数声明为一个异步函数。异步函数内部可以使用await关键字来等待一个异步操作的完成,并且返回一个Promise对象。 使用async/await可以避免使用回调函数、嵌套的Promise链和复杂的错误处理。通过async/await,代码可以按照线性的方式书写,并且保持原有的控制结构。 在使用async/await时,可以使用try/catch块来捕获异步操作中的错误,这样就可以像处理同步操作的错误一样处理异步操作的错误。 一个典型的async/await的用法是在异步函数内部使用await关键字等待另一个异步操作的完成,然后使用得到的结果进行后续的处理。在等待异步操作时,代码会暂停执行,而不会阻塞线程。 总之,JavaScriptasync/await是一种更简洁、易读且易于维护的处理异步操作的编程模式。它提供了一种能够让开发者以线性的方式编写异步代码的方式,并且能够更好地处理异步操作中的错误。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值