I am trying to do task step by step.
I have a for loop in a method:
async changeTimeStep3() {
for (let i = 1; i < 10; i++) {
await this.do(i)
}
}
for each step must do() task.
do(i) {
this.http
.getDataFromServer(
"api/example?id=" +i
)
.subscribe((response) => {
console.log(i);
});
}
I want to wait to get response and after response ing go to next i
But not work console.log print:
2
3
5
1
4
7
8
9
6
Note time to receive response from api is not fix.
Any help?
I am trying to do task step by step.
I have a for loop in a method:
async changeTimeStep3() {
for (let i = 1; i < 10; i++) {
await this.do(i)
}
}
for each step must do() task.
do(i) {
this.http
.getDataFromServer(
"api/example?id=" +i
)
.subscribe((response) => {
console.log(i);
});
}
I want to wait to get response and after response ing go to next i
But not work console.log print:
2
3
5
1
4
7
8
9
6
Note time to receive response from api is not fix.
Any help?
Share edited Jul 26, 2022 at 8:00 Mahdi Zarei 7,5068 gold badges32 silver badges64 bronze badges asked Dec 13, 2021 at 13:48 jack rosjack ros 291 silver badge8 bronze badges 4- 1 do doesn't return a promise to await... – jonrsharpe Commented Dec 13, 2021 at 13:51
- why? and how to do in right way? @jonrsharpe – jack ros Commented Dec 13, 2021 at 13:54
- I don't know why it doesn't return a promise, you presumably wrote it like that, or why you're awaiting it even though it doesn't. And if you want to await it, try actually returning a promise that resolves when the work is done (or rejects if it fails). – jonrsharpe Commented Dec 13, 2021 at 13:57
- I realize that this doesn't answer your question, but making calls like that is an anti-pattern. If you need to await multiple calls in a loop, you need to pass a collection to the query. – VSO Commented Dec 13, 2021 at 13:57
3 Answers
Reset to default 8To get your loop print out the values in the correct order you could transform your observables into promises, so your await
will block the for
-loop as long as the HTTP call hasn't resolved.
import { firstValueFrom } from 'rxjs';
...
async do(i: number) {
await firstValueFrom(
this.http
.getDataFromServer(
"api/example?id=" +i
)
);
console.log(i);
}
Using firstValueFrom
or lastValueFrom
are the RxJS ways to get a promise as return value from an observable (since toPromise
got deprecated, you can read more about that here).
You can return a Promise
and use resolve()
in response part of your ajax. Like:
do(i) {
return new Promise((resolve, reject) => {
this.http.getDataFromServer("api/example?id=" +i).subscribe((response) => {
resolve(response);
}, (error) => {
console.error(error);
reject();
});
});
}
The problem here is that do is not an asynchronous method; it is a synchronous method that triggers an asynchronous one. As such, the await pletes immediately.
You may be able to do this:
do(i) {
return firstValueFrom(this.http
.getDataFromServer(
"api/example?id=" +i
))
}
It converts the observable into a promise and returns it. So, now the do() method returns an async object. To output from the do method; you can use a pipe:
return firstValueFrom(this.http
.getDataFromServer(
"api/example?id=" +i
).pipe(tap((result) => {console.log(i);}))
All that said, using async/await in Angular is unusual. An RXJS operator is usually considered the right approach. Possibly with concat