• Najnowsze pytania
  • Bez odpowiedzi
  • Zadaj pytanie
  • Kategorie
  • Tagi
  • Zdobyte punkty
  • Ekipa ninja
  • IRC
  • FAQ
  • Regulamin
  • Książki warte uwagi

Problem z asynchronicznością Node

Aruba Cloud PRO i VPS, Openstack, VMWare, MS Hyper-V
0 głosów
125 wizyt
pytanie zadane 12 lipca 2021 w JavaScript przez KopfSzmercen Bywalec (2,870 p.)

Uczę się Node i piszę pierwszą małą apkę. Mam problem z jedną funkcją, nie bardzo wiem co może być powodem tego że nie działa tak jak chcę, przyglądam się jej już dość długo i szczerze im dłużej to robię tym bardziej sam sobie to komplikuję. Teraz do rzeczy: pobieram z bazy danych użytkownika, który ma tablicę z id jego tasków. Potem chcę pobrać z bazy te taski po ich _id. Więc lecę po tablicy i daję zapytanie do bazy. Mimo zastosowania funkcji asynchronicznej i użycie await działanie 'przechodzi' dalej nie czekając na zapytanie, renderuje widok bez poczekania na odpowiedź z bazy, ale gdy taski zostaną już dociągnięte z bazy w console logu mam poprawne dane. 

export const getMainPage = async (
  req: express.Request,
  res: express.Response
) => {
  const userTasks: any =  [];
  
  try {
    const fetchedUser = await User.findOne({ _id: req.session.loggedUserId });
  
    await fetchedUser.tasks.forEach(async (id: string) => {
      const fetchedTask = await Task.findOne({ _id: id });
      userTasks.push(fetchedTask);
      // console.log(userTasks); tutaj dostaje poprawne dane
    });
    
    //console.log(userTasks); ale dopiero po tym logu i po renderze widoku
    res.render("main-page", {
      path: "main-page",
      csrfToken: req.csrfToken(),
      tasks: userTasks,
    });

  } catch (e) { 
    console.log(e);
    res.redirect('/add-task');
  }
};

Dzięki z góry za poświęcony czas.

2 odpowiedzi

+3 głosów
odpowiedź 12 lipca 2021 przez ScriptyChris Mędrzec (190,440 p.)
wybrane 13 lipca 2021 przez KopfSzmercen
 
Najlepsza

Do forEach przekazujesz asynchroniczną funkcję. Tam await działa, ale sam forEach zwraca undefined, więc jego await rozwiązuje się od razu zamiast poczekać na promisy z wnętrza forEach. Zamień forEach na map i wrzuć to w Promise.all - przed którym użyj await. Wtedy poczekasz na rozwiązanie wszystkich promisów z tablicy zwróconej przez map.

komentarz 13 lipca 2021 przez KopfSzmercen Bywalec (2,870 p.)
Dzięki za wytłumaczenie :)
komentarz 13 lipca 2021 przez Wiciorny Ekspert (245,430 p.)
też się doedukowałem, bo ja w bebechach siedze ;D zwykle.
+1 głos
odpowiedź 12 lipca 2021 przez Wiciorny Ekspert (245,430 p.)

Więc lecę po tablicy i daję zapytanie do bazy. Mimo zastosowania funkcji asynchronicznej i użycie await działanie 'przechodzi' dalej nie czekając na zapytanie

 zwraca Ci obietnica https://developer.mozilla.org/pl/docs/Web/JavaScript/Reference/Statements/async_function

Musisz obsłużyć zwróconą obietnice. https://developer.mozilla.org/en-US/docs/Learn/JavaScript/Asynchronous/Async_await

Podobne pytania

+2 głosów
2 odpowiedzi 126 wizyt
0 głosów
1 odpowiedź 114 wizyt
pytanie zadane 18 sierpnia 2020 w JavaScript przez michael.scott Nowicjusz (200 p.)
0 głosów
0 odpowiedzi 128 wizyt

90,303 zapytań

138,899 odpowiedzi

311,088 komentarzy

60,019 pasjonatów

Motyw:

Akcja Pajacyk

Pajacyk od wielu lat dożywia dzieci. Pomóż klikając w zielony brzuszek na stronie. Dziękujemy! ♡

Sklep oferujący ćwiczenia JavaScript, PHP, rozmowy rekrutacyjne dla programistów i inne materiały

Oto dwie polecane książki warte uwagi. Pełną listę znajdziesz tutaj.

...