JavaScript as a language is heavily asynchronous, with promises being deeply integrated. The inclusion of async/await syntax has massively simplified this, making asynchronous code much more readable and much quicker to write. Being able to mark methods as async makes it much easier to integrate into existing parts of code, sometimes causing large chains of method calls to become async for a single deep method call.
This is a problem that’s been referred to as “function colours” in the past, and basically means you’re segmenting what code can be run from where. In the allegory blue functions are normal functions, while red ones are async. You can call blue functions from red functions, but not red functions from blue functions. Therefore, if you need to call a red function from a blue one, it must become red, which then affects the entire function tree. Making one function async might require making ten functions async, significantly compounding the problem. While this is sometimes the best solution, can the overhead of these promises pose a problem for hot code?
Benchmarks
As of writing, the v8 JavaScript engine that powers both Chromium and NodeJS does not optimise out redundant promises. Due to this, a decent benchmark to determine just the overhead of promises is to take a typical function call and make it a promise. For this case, I'm using a function that calculates a position in the Fibonacci sequence. These tests are intentionally purely CPU-bound, to correctly emulate functions that unnecessarily are wrapped inside promises.
Test 1 - Recursive Fibonacci
The first test is using a recursive Fibonacci function. Using a recursive function as a test case here means we will get a compounding overhead from the promises. This result will show us a relatively worse case example.
The following code has been used,
// Normal case
function fibonacci(num) {
if (num <= 1) return 1;
return fibonacci(num - 1) + fibonacci(num - 2);
}
for (let i = 0; i < 10; i++) {
fibonacci(i);
}
// Promise case
async function fibonacci(num) {
if (num <= 1) return 1;
return (await fibonacci(num - 1)) + (await fibonacci(num - 2));
}
for (let i = 0; i < 10; i++) {
// The site does not allow top-level await calls, however it waits for all
// promises to resolve before completing, making this relatively okay to do.
fibonacci(i).then(() => {});
}
When running this on JSBench.Me, the promise case is shown to be 86% slower on Chrome 87 than the non-promise case.
While this is an unrealistic case, it shows that promises can significantly impact performance, especially in hot code paths.
Coming back to this 4 years later and running on Chrome 131 shows that it’s now 91% slower. This is likely due to optimisations in v8 affecting the non-promise pathway more than the promise one, rather than promises actually getting slower over time. This demonstrates that at least in the past 4 years, this overhead has not improved.
Test 2 - Non-Recursive Fibonacci
A non-recursive Fibonacci sequence shows what sort of overhead a low-cost method call could have to get a more real-world view. In this case, each call to the function will only have the overhead of a single promise.
The following code has been used,
// Normal case
function fibonacci(num) {
var a = 1,
b = 0,
temp;
while (num >= 0) {
temp = a;
a = a + b;
b = temp;
num--;
}
return b;
}
for (let i = 0; i < 100; i++) {
fibonacci(i);
}
// Promise case
async function fibonacci(num) {
var a = 1,
b = 0,
temp;
while (num >= 0) {
temp = a;
a = a + b;
b = temp;
num--;
}
return b;
}
for (let i = 0; i < 100; i++) {
fibonacci(i).then(() => {});
}
When running this on JSBench.Me, the promise case is shown to be 81% slower on Chrome 87 than the non-promise case.
This result shows that adding an async modifier to the method can almost double the time it takes to execute simple functions. For hot code, this could make a significant difference.
Similarly on Chrome 131, this case is 90% slower. This again appears to be due to further optimisations on the non-promise pathway, rather than promises getting slower. The gap widens, but promises aren’t getting worse, non-promises are getting better.
Test 3 - Promise vs Async/Await
So we've seen what just adding the async keyword can do, but what if we use promises directly instead?
Using async code from the previous test case, and the following code,
// Promise case
function fibonacci(num) {
return new Promise((resolve, reject) => {
var a = 1,
b = 0,
temp;
while (num >= 0) {
temp = a;
a = a + b;
b = temp;
num--;
}
resolve(b);
});
}
for (let i = 0; i < 100; i++) {
fibonacci(i).then(() => {});
}
When running this on JSBench.Me, the case using promises directly is shown to be a further 26% slower on Chrome 87 than the async/await case.
On Chrome 131 this test is more interesting, as the promise case is now only 15% slower. This indicates that the gap between directly using promises or async/await is slowly closing.
Solutions
There are a few potential solutions for this issue. Firstly, however, it's important to point out that unless you've confirmed a section of code to be causing performance issues, you don't need to optimise it. Once you've used a profiler to verify that it's worth optimising a piece of code, then it's worth investigating.
The simplest solution here is to perform data fetching or other asynchronous operations closer to the application's root and pass the resulting data down. Often program structures that involve deep-nested async/await paths exhibit poor separation of concerns. Ideally, a single system should not load and use data; instead, it should receive the data from another system that loads it. This structure also has the added benefit of being much more testable. Doing this should prevent the need to nest async/await methods deeply.
Another option is to only use the async keyword for the inner-most method if possible. Each method with an async keyword adds overhead, so if you can work with the promise directly outside of that function, you're not introducing further overhead. This technique can be even more helpful when the method only sometimes needs to return a promise. In this case, including the async keyword introduces the overhead always. If you return a promise directly, overhead (albeit slightly more overhead) happens only a portion of the time.
Conclusion
Promises and Async/Await introduce measurable overhead into JavaScript code. This problem can be worked around in most cases when necessary, yet, it will not pose any issue most of the time. As with most things, understanding the various trade-offs in performance is essential when optimising code. If you've identified a hot code path that makes heavy use of async/await, it may be worth investigating ways to minimise that.
Hi, I'm Maddy Miller, a Senior Software Engineer at Clipchamp at Microsoft. In my spare time I love writing articles, and I also develop the Minecraft mods WorldEdit, WorldGuard, and CraftBook. My opinions are my own and do not represent those of my employer in any capacity. Find out more.