blog-web/source/_posts/JavaScript/优雅写异步与循环.md
2019-07-01 00:12:32 +08:00

82 lines
2.1 KiB
Markdown

---
title: 优雅写异步与循环
date: 2019-06-29 23:39:46
tags:
- JavaScript
categories:
- JavaScript
---
JS当中的循环都是非异步的
包括但不限于
1. Array.prototype中的`forEach`
2. `for ... in` 语法
3. `for ... of` 语法
<!-- more -->
所以如果要在循环内的异步全部完成后做某些事情
例如
```javascript
// 这里我只是简单构造了一个异步
// 实际运用当中譬如数据库查询、文件读写等操作, 通常都是异步的
function show(num) {
Promise.resolve(num).then(console.log)
}
const arr = [100, 200, 300]
console.log('start')
arr.forEach(show)
console.log('end')
```
上面的写法根据事件队列的机制, 显然会先输出end, 再输出数组元素的值
当然我们可以在Promise的resolve函数当中判断是否到达了数组最后一个元素, 把输出end的操作写进resolve函数里面
显然这不够优雅, 而且很多时候也不方便这样做
实现方式
#### Promise.all
```javascript
const arr = [100, 200, 300]
console.log('start')
console.time('promise all in')
Promise.all(arr.map(show)).then(() => {
console.timeEnd('promise all in')
console.log('end')
})
```
为了比较执行性能的差异, 加了一个计时 ( Nodejs环境运行 )
![Promise.all时间](/images/JavaScript/Promise.all时间.png)
#### async/await
```javascript
const arr = [100, 200, 300];
(async function() {
console.log('start')
console.time('await all in')
for await (let i of arr.map(show)) {}
console.timeEnd('await all in')
console.log('end')
})()
```
![await时间](/images/JavaScript/await时间.png)
由于await必须用在async修饰的函数当中, 所以包装了一层
实际执行时间与Promise.all差不多
### One by one
这种方式效率最低,有点类似于同步语言中的循环,一个接着一个执行,耗时自然也就是所有异步方法耗时的总和。对资源的消耗最小。
```javascript
const arr = [100, 200, 300];
(async function() {
console.log('start')
console.time('await one by one')
for (let item of arr) {
await show(item)
}
console.timeEnd('await one by one')
console.log('end')
})()
```