170 похожих чатов

Всем доброй ночи! Прошу помощи с циклом событий в ноде. Скорее всего,

вопрос базового уровня, но не стал писать в js_noobs, т.к. вопрос относится в том числе к работе event loop'а в ноде.

function wrapper() {
console.log(1);
Promise.resolve().then(() => console.log('Promise handler!'));
console.log(3);
}

function someHardWork() {
let result = [];
for (let i = 0; i < 1e7; i++)
result.push(i * i * i);
return result.reduce((prev, current) => prev + current);
}

console.log(4);
wrapper();
console.log(5);
console.log(someHardWork());

Output:
4
1
3
5
2.499999499999934e+27
Promise handler!

Если я правильно понял механизм работы цикла событий в V8, то он закидывает на выполнение микротаски сразу же после того, как текущий стек вызовов становится пустым.
И в этом случае цикл событий должен был закинуть на выполнение коллбэк из промиса из wrapper() сразу же после возврата из wrapper().
Но этого не происходит, а происходит выполнение всего синхронного кода (включая длительную задачу в someHardWork()), и только потом выполняется микротаск (коллбэк промиса).

Получается, что в любом случае сначала выполняется весь синхронный код верхнего уровня в модуле, и только потом выполняются микротаски, включая nextTick() и промисы?

2 ответов

8 просмотров

Всё выполняется, как и должно. У тебя сперва запускается сам скрипт. По сути это одна общая функция колбэк, чтобы данные были консистентными, код функции должен быть исполнен до конца. Эта функция формирует первую запись в стеке вызовов, затем на стек бросаются вызовы вложенных функций. Во вложенной функции ты добавил в microtask queue свой promise callback, его очередь как раз после основной функции (глобального скрипта, так сказать). Это немного упрощенная картина, ещё там присутствует ряд фаз и свои правила их исполнения, но это другая история.

Похожие вопросы

Обсуждают сегодня

Какой-то там пердун в 90-х решил, что есть какая-то разная типизация. Кого вообще это волнует?
КТ315
49
void terminal_scroll() { memmove(terminal_buffer, terminal_buffer + VGA_WIDTH, buffer_size - VGA_WIDTH); memset(terminal_buffer + buffer_size - VGA_WIDTH, 0, VGA_WIDTH); ...
Егор
47
Всем привет! Подскажите, пожалуйста, в чем ошибка? Настраиваю подключение к MySQL. Либы лежат рядом с exe. Все как по "учебнику"
Евгений
16
А можете как-то проверить меня по знаниям по ассемблеру?
A A
132
Здравствуйте! У меня появилась возможность купить книгу "Изучай Haskell во имя добра!". Но я где-то слышал, что эта книга устарела. Насколько это правда??
E
22
Здравствуйте! Я вот на stepic решаю задачи на хаскеле https://stepik.org/lesson/8443/step/8?unit=1578 мой код import Data.List (isInfixOf) removing :: String -> [String] ->...
E
10
Камрады, кто тесно работал с vtv, хотел уточнить. Ширина column задаётся жёстко на этапе создания дерева или можно в рантайме ее менять программно (не мышкой)?
Ed Doc
10
да ладно ... что там неочевидного ? глянуть в исх-ки датасета и/или кверика чтобы понять в каком месте и как выполняется обращения к св-вам blablaSQL - минутное дело, даже е...
Сергей
7
Здесь для arm кто-нибудь кодит ?
Nothing
52
Всем привет, у меня есть сервер принимающий входящие HTTP подключения, как проверить, что подключение было через прокси или нет, есть какие то поля в заголовках по которым мо...
Кибер Бомж
8
Карта сайта