I am using async
/await
in several places in my code.
For example, if I have this function:
async function func(x) {
...
return y;
}
Then I always call it as follows:
async function func2(x) {
let y = await func(x);
...
}
I have noticed that in some cases, I can omit the await
and the program will still run correctly, so I cannot quite figure out when I must use await
and when I can drop it.
I have concluded that it is "legitimate" to drop the await
only directly within a return statement.
For example:
async function func2(x) {
...
return func(x); // instead of return await func(x);
}
Is this conclusion correct, or else, what am I missing here?
EDIT:
A small (but important) notion that has not been mentioned in any of the answers below, which I have just encountered and realized:
It is NOT "legitimate" to drop the await
within a return
statement, if the called function may throw an exception, and that statement is therefore executed inside a try
block.
For example, removing the await
in the code below is "dangerous":
async function func1() {
try {
return await func2();
}
catch (error) {
return something_else;
}
}
The reason is that the try
block pletes without an exception, and the Promise
object returns "normally". In any function which calls the outer function, however, when this Promise
object is "executed", the actual error will occur and an exception will be thrown. This exception will be handled successfully in the outer function only if await
is used. Otherwise, that responsibility goes up, where an additional try
/catch
clause will be required.
I am using async
/await
in several places in my code.
For example, if I have this function:
async function func(x) {
...
return y;
}
Then I always call it as follows:
async function func2(x) {
let y = await func(x);
...
}
I have noticed that in some cases, I can omit the await
and the program will still run correctly, so I cannot quite figure out when I must use await
and when I can drop it.
I have concluded that it is "legitimate" to drop the await
only directly within a return statement.
For example:
async function func2(x) {
...
return func(x); // instead of return await func(x);
}
Is this conclusion correct, or else, what am I missing here?
EDIT:
A small (but important) notion that has not been mentioned in any of the answers below, which I have just encountered and realized:
It is NOT "legitimate" to drop the await
within a return
statement, if the called function may throw an exception, and that statement is therefore executed inside a try
block.
For example, removing the await
in the code below is "dangerous":
async function func1() {
try {
return await func2();
}
catch (error) {
return something_else;
}
}
The reason is that the try
block pletes without an exception, and the Promise
object returns "normally". In any function which calls the outer function, however, when this Promise
object is "executed", the actual error will occur and an exception will be thrown. This exception will be handled successfully in the outer function only if await
is used. Otherwise, that responsibility goes up, where an additional try
/catch
clause will be required.
- Example when you have big data and dont want to lose information you need to use the await, because general operation close the connection without the information get plete – HudsonPH Commented Nov 21, 2017 at 8:48
- Another example is using api, you need to wait the server response so you need to use await to dont lose information – HudsonPH Commented Nov 21, 2017 at 8:49
4 Answers
Reset to default 8If func
is an async function then calling it with and without await
has different effects.
async function func(x) {
return x;
}
let y = await func(1); // 1
let z = func(1) // Promise (resolves to 1)
It is always legitimate to omit the await
keyword, but means you will have to handle the promises in the traditional style instead (defeating the point of async/await in the first place).
func(1).then(z => /* use z here */)
If your return statements use await
then you can be sure that if it throws an error it can be caught inside your function, rather than by the code that calls it.
await
just lets you to treat promises as values, when used inside an async
function.
On the other hand, async
works quite the opposite, it tags the function to return a promise, even if it happens to return a real, synchronous value (which sounds quite strange for an async function... but happens often when you have a function that either return a value or a promise based on conditions).
So:
I have concluded that it is "legitimate" to drop the await only directly within a return statement.
In the last return
statement of an async
function, you just are returning a Promise, either you are return actually a directly a promise, a real value, or a Promise-as-value with the await
keyword.
So, is pretty redundant to use await
in the return statement: you're using await
to cast the promise to a value -in the context of that async execution-, but then the async
tag of the function will treat that value as a promise.
So yes, is always safe to drop await
in the last return statement.
PS: actually, await expects any thenable, i.e. an object that has a then
property: it doesn't need a fully spec pliant Promise to work, afaik.
PS2: of course, you can always drop await
keyword when invoking synchronous functions: it isn't needed at all.
An async function always returns a Promise.
So please keep in mind that these writing of an async function are all the same:
// tedious, sometimes necessary
async function foo() {
return new Promise((resolve) => resolve(1)))
}
// shorter
async function foo() {
return Promise.resolve(1)
}
// very concise but calling `foo` still returns a promise
async function foo() {
return 1 // yes this is still a promise
}
You call all of them via foo().then(console.log)
to print 1
. Or you could call them from another async function via await foo()
, yet it is not always necessary to await the promise right away.
As pointed out by other answers, await
resolves the promise to the actual return value statement on success (or will throw an exception on fail), whereas without await
you get back only a pending promise instance that either might succeed or fail in the future.
Another use case of omitting (i.e.: being careful about its usage) await
is that you might most likely want to parallelize tasks when writing async code. await
can hinder you here.
Compare these two examples within the scope of an async function:
async function func() {
const foo = await tediousLongProcess("foo") // wait until promise is resolved
const bar = await tediousLongProcess("bar") // wait until promise is resolved
return Promise.resolve([foo, bar]) // Now the Promise of `func` is marked as a success. Keep in mind that `Promise.resolve` is not necessary, `return [foo, bar]` suffices. And also keep in mind that an async function *always* returns a Promise.
}
with:
async function func() {
promises = [tediousLongProcess("foo"), tediousLongProcess("bar")]
return Promise.all(promises) // returns a promise on success you have its values in order
}
The first will take significantly longer than the last one, as each await
as the name implies will stop the execution until you resolve the first promise, then the next one.
In the second example, the Promise.all
the promises will be pending at the same time and resolve whatever order, the result will then be ordered once all the promises have been resolved.
(The Bluebird promise library also provides a nice Bluebird.map
function where you can define the concurrency as Promise.all
might cripple your system.)
I only use await
when want to work on the actual values. If I want just a promise, there is no need to await its values, and in some cases it may actually harm your code's performance.
I got a good answer above, here is just another explanation which has occurred to me.
Suppose I have this:
async function func(x) {
...
return y;
}
async function func2(x) {
...
return await func(x);
}
async function func3(x) {
let y = await func2(x);
...
}
The reason why I can safely remove the await
in the return statement on func2
, is that I already have an await
when I call func2
in func3
.
So essentially, in func3
above I have something like await await func(x)
.
Of course, there is no harm in that, so it's probably better to keep the await
in order to ensure desired operation.