вопрос базового уровня, но не стал писать в 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() и промисы?
Конечно
Всё выполняется, как и должно. У тебя сперва запускается сам скрипт. По сути это одна общая функция колбэк, чтобы данные были консистентными, код функции должен быть исполнен до конца. Эта функция формирует первую запись в стеке вызовов, затем на стек бросаются вызовы вложенных функций. Во вложенной функции ты добавил в microtask queue свой promise callback, его очередь как раз после основной функции (глобального скрипта, так сказать). Это немного упрощенная картина, ещё там присутствует ряд фаз и свои правила их исполнения, но это другая история.
Обсуждают сегодня