JavaScript Async and await in Loops

preface

In my recent project, I met a requirement of batch application. At that time, there was only interface for single application, so I came up with a solution of loop array request interface, and then ENCOUNTERED problems of async/await and loop. I found that the use of async/await in forEach did not take effect, so I found the problem in the Process of Google. This article explains in detail, and the case is complete and easy to understand. It is a rare and good article, so I translated it for your reference, you can discuss in the comments section if you have any questions!

Oh? How did I end up doing that, you ask? The backend students gave me a batch application interface.

The body of the

Basic async and await are relatively simple to use, and things get a little more complicated when you try to use await in a loop.

case

For example, let’s say you want to know the number of fruits in the fruitBasket.

const fruitBasket = {
    apple: 27.grape: 0.pear: 14
}
Copy the code

You want to get the number of each fruit in the basket. To get them, you can define a getNumFruit function.

const getNumFruit = fruit= > {
    return fruitBasket[fruit]
}
const numApples = getNumFruit('apple')
console.log(numApples)	/ / 27
Copy the code

Now, say fruitBasket is on a remote server. It takes a second to access it. We can use a timeout timer to simulate this one-second delay.

const sleep = ms= > {
    return new Promise(resolve= > setTimeout(resolve, ms))
}
const getNumFruit = fruit= > {
    return sleep(1000).then(v= > fruitBasket[fruit])
}
getNumFruit('apple')
	.then(num= > console.log(num)) / / 27
Copy the code

Suppose you don’t want to use Promise to perform asynchronous tasks, and you want to use async/await to perform asynchronous tasks synchronously, as follows:

const control = async_ = > {console.log('Start')
    
    const numApples = await getNumFruit('apple');
    console.log(numApples);
    
    const numGrapes = await getNumFruit('grape');
    console.log(numGrapes);
    
    const numPears = await getNumFruit('pear');
    console.log(numPears);
    
    console.log('End')}Copy the code

Use Await in the for loop

Suppose we define an array of fruits.

const fruitsToGet = ['apple'.'grape'.'pear']
Copy the code

Loop through the array

const forLoop = async_ = > {console.log('Start')
    
    for(let index = 0; index < fruitsToGet.length; index++) {
        // Get num of each fruit
    }
    
    console.log('End')}Copy the code

In this for loop, we’ll use getNumFruit to get and print the number of each fruit.

Because getNumFruit returns a Promise, we wait for the resolved result to return before printing.

const forLoop = async_ = > {console.log('Start')
    
    for (let index = 0; index < fruitsToGet.length; index ++) {
        const fruit = fruitsToGet[index]
        const numFruit = await getNumFruit(fruit)
        console.log(numFruit)
    }
    
    console.log('End')}Copy the code

When you use await, you might expect JavaScript to pause execution until a promise returns. This means that await should be executed sequentially in a for loop

And the results are exactly what you’d expect:

'Start'
'Apple: 27'
'Grape: 0'
'Pear: 14'
'End'
Copy the code

This behavior works in most loops (like while and for of loops)…

But it can’t handle loops that need to be called back. For example, forEach, Map, Filter, and Reduce. In the next few sections, we’ll look at how await affects forEach, Map, and filter.

Use await in forEach loop

As in the previous example, we first iterate through the fruit array.

const forEachLoop = _= > {
    console.log('Start')
    
    fruitsToGet.forEach(fruit= > {
        // Send a promise for each fruit
    })
    
    console.log('End')}Copy the code

Then we tried using getNumFruit to get the fruit count. (Note the async keyword in the callback function, we need this async because await is in the callback).

const forEachLoop = _= > {
    console.log('Start')
    
    fruitsToGet.forEach(async fruit => {
        const numFruit = await getNumFruit(fruit)
        console.log(numFruit)
    })
    
    console.log('End')}Copy the code

You might expect the console to print something like this:

'Start'
'27'
'0'
'14'
'End'
Copy the code

Instead, JavaScript calls console.log(‘End’) before the promise in the forEach loop gets the result.

'Start'
'End'
'27'
'0'
'14'
Copy the code

The reason is simple: forEach only supports synchronous code.

Take a look at the Polyfill version of forEach, which looks like this pseudo-code after simplification.

while (index < arr.length) {
  callback(item, index)   // This is the callback we passed in
}
Copy the code

As you can see from the above code, forEach simply executes the next callback and does not handle the asynchronous case. And if you use break in the callback, it doesn’t end the loop.

Why the for… “Of” inside would make “await” valid.

Because of the for… The internal processing mechanism of of is different from that of forEach, which calls the callback directly, for… Of is traversed by iterator.

Use await in map

If you use await in a map, the map will always return a Promise array.

const mapLoop = async_ = > {console.log('Start')
    
    const numFruits = await fruitsToGet.map(async fruit => {
        const numFruit = await getNumFruit(fruit)
        return numFruit
    })
    
    console.log(numFruits)
    console.log('End')}Copy the code
'Start'
'[Promise, Promise, Promise]'
'End'
Copy the code

If you use await in map, map always returns Promises, you have to wait for Promises array to be processed. Or do this by await promise. all(arrayOfPromises).

const mapLoop = async_ = > {console.log('Start')
    
    const promises = fruitsToGet.map(async fruit => {
        const numFruit = await getNumFruit(fruit)
        return numFruit
    })
    
    const numFruits = await Promise.all(promises);
    console.log(numFruits);
    
    console.log('End')}Copy the code

The running results are as follows:

'Start'
'[27, 0, 14]'
'End'
Copy the code

If you wish, you can handle the return value in the Promise, and the parsed value will be the returned value.

const mapLoop = async_ = > {// ...
  const promises = fruitsToGet.map(async fruit => {
    const numFruit = await getNumFruit(fruit)
    // Adds onn fruits before returning
    return numFruit + 100
  })
  // ...
}
Copy the code
'Start'
'[127, 100, 114]
'End'
Copy the code

Use await in filter loop

When you use filter, you want to filter an array with a particular result. Assume that the number of arrays filtered is greater than 20.

If you normally use filter (without await), it would look like this:

const filterLoop = _= > {
    console.log('Start')
    
    const moreThan20 = fruitsToGet.filter(fruit= > {
        const numFruit = fruitBasket[fruit]
        return numFruit > 20
    })
    
    console.log(moreThan20)
    console.log('End')}Copy the code
Start
["apple"]
END
Copy the code

Await in filter doesn’t work in the same way, in fact it doesn’t work at all and you get an unfiltered array.

const filterLoop = async_ = > {console.log('Start')
    
    const moreThan20 = await fruitsToGet.filter(async fruit => {
        const numFruit = await getNumFruit(fruit)
        return numFruit > 20
    })
    
    console.log(moreThan20)
    console.log('End')}Copy the code
'Start'
['apple'.'grape'.'pear']
'End'
Copy the code

Why is that?

When you use await in a filter callback, the callback always returns a promise. Because Promises are always true, everything in the array goes through filter. Use the code below the await class in filter

const filtered = array.filter((a)= > true)
Copy the code

The correct three steps to use await in filter

  1. Use map to return a Promise array
  2. Await the result of processing with await
  3. Use filter to process the result returned
const filterLoop = async_ = > {console.log('Start')
    const promises = await fruitsToGet.map(fruit= > getNumFruit(fruit))
    const numFruits = await Promise.all(promises)
    const moreThan20 = fruitsToGet.filter((fruit, index) = > {
        const numFruit = numFruits[index]
        return numFruit > 20
    })
    
    console.log(moreThan20)
    console.log('End')}Copy the code
Start
[ 'apple' ]
End
Copy the code

Use await in reduce

If you want to count the total number of fruits in fruitBastet. The reduce loop is typically used to iterate through a set of numbers and add the numbers.

const reduceLoop = _= > {
  console.log('Start');

  const sum = fruitsToGet.reduce((sum, fruit) = > {
    const numFruit = fruitBasket[fruit];
    return sum + numFruit;
  }, 0)

  console.log(sum)
  console.log('End')}Copy the code

When you use await in reduce, the result becomes very confusing.

const reduceLoop = async_ = > {console.log('Start')

  const sum = await fruitsToGet.reduce(async (sum, fruit) => {
    const numFruit = await getNumFruit(fruit)
    return sum + numFruit
  }, 0)

  console.log(sum)
  console.log('End')}Copy the code
'Start'
'[object Promise]14'
'End'
Copy the code

What the hell is 14?

It’s interesting to dissect that.

  1. In the first iteration, sum is 0. NumFruit is 27(the value obtained from getNumFruit(apple)), 0 + 27 = 27.
  2. In the second iteration, sum is a promise. B: Why? Because asynchronous functions always return Promises! NumFruit is 0. A promise cannot be added to an object properly, so JavaScript converts it to an [Object Promise] string. [Object Promise] + 0 is object Promise] 0.
  3. On the third iteration, sum is also a promise. NumFruit is 14. [object Promise] + 14 is [Object Promise] 14.

This means that you can use await in reduce callbacks, but you must remember to wait for the accumulator first!

const reduceLoop = async_ = > {console.log('Start');

  const sum = await fruitsToGet.reduce(async (promisedSum, fruit) => {
    const sum = await promisedSum;
    const numFruit = await fruitBasket[fruit];
    return sum + numFruit;
  }, 0)

  console.log(sum)
  console.log('End')}Copy the code

But as you can see in the figure above, await operations take a long time. This happens because the reduceLoop waits for each iteration to complete the promisedSum.

One way to speed up the Reduce cycle is to reduce oop in a second if you wait for getNumFruits() before waiting for promisedSum:

const reduceLoop = async_ = > {console.log('Start');

  const sum = await fruitsToGet.reduce(async (promisedSum, fruit) => {
    const numFruit = await fruitBasket[fruit];
    const sum = await promisedSum;
    return sum + numFruit;
  }, 0)

  console.log(sum)
  console.log('End')}Copy the code

This is because Reduce can trigger all three getNumFruit Promises before waiting for the next iteration of the loop. However, this method is a bit confusing because you have to pay attention to the order in which you wait.

The simplest (and most efficient) way to use Wait in Reduce is

  1. Use map to return a Promise array
  2. Await the result of processing with await
  3. The returned results are processed using reduce
const reduceLoop = async_ = > {console.log('Start')

  const promises = fruitsToGet.map(getNumFruit)
  const numFruits = await Promise.all(promises)
  const sum = numFruits.reduce((sum, fruit) = > sum + fruit)

  console.log(sum)
  console.log('End')}Copy the code

This version is easy to read and understand and takes a second to count the fruit.

What do you see up there

  1. If you want to continuously perform await calls, use loops without callbacks (for… Of, for loop, while loop)
  2. Never use await with forEach
  3. Do not use await in Filter and reduce. If necessary, further processing with Map and then processing with Filter and reduce.

Reference: why await cannot be used in forEach