потому что запустить консольлог и запланировать запуск консольлога через 1.5 сек это разное. и поведение вар/лет разное.
вот можно поподробнее про работу вар и лет в данном случае
потомучто цикл уже оттарабанил к тому моменту, как вызываются колбеки из сеттаймаута. В случае с var - так как hoisting + функциональная область видимости - var обявляется каждый тик во внешнем скоупе цикла и потому перезаписывается. В случае с let - блочная область - каждую итерацию внутри тела цикла создается переменная в своем скоупе. И когда доходит то вызова колбека - они выводят каждый свою переменную i
так. с var понятно. а вот с let Нет. в моем понимании у let всегда должно быть тоже 6, т.к settimeout асинхронная операция и она работает вне синхронного кода. тоесть фор синхронно отработал до 6 и вышел. а потом показался последний let i = 6.
так почти и есть, к моменту вызова колбеков из setTimeout в случае с let цикл уже давно оттарабанил и все переменные i уже обьявлены. Но в отличие от вар = их 6 штук каждая в своем скоупе, в котором вызывается колбек таймаута, а не 1 во внешнем скоупе 6 раз перезаписанная. Нужно про области видимости\замыкания почитать
а. то есть for замыкает settimeout верно?
в блочном скоупе да, каждую итерацию создается свой скоуп внутри блока, который имеет доступ к внешнему, типо 10 раз проитерировался - получил 10 слоев контекста внутри блока и колбеки вызываются из этого контекста. Но в случае с вар - они там не находят эту переменную изза всплытия и ищут во внешнем скоупе (вне блока) - а она там 10 раз переопределена и имеет значение последнего тика
интересно. тоесть правильно я понимаю что если осталась какая-то работа после выполнения синхронного стека, то v8 его выполняет? верно?
Я не понял вопрос :)
ну допустим я такой код запускаю в ноде. я думал что в логах появится только done и все. но потом появляются остальные цифры, я думал что выполняется синхронный цикли в все нода останавливается, но она продолжает что-то делать. for(let i = 0; i<=6; i ++) { setTimeout(()=> console.log(i), 1000) } console.log("done")
Ты из основного стека в процессе итерации - закинул 6 колбеков (вернее 6 сеттаймаутов) в браузерный апи, который прокрутил таймеры и поместил эти колбеки в очередь колбеков. Евент луп же преджде чем взяться за очередь колбеков - ждёт когда освободится основной стек (микротаски пока опустим), а он освободится в твоем примере, когда consolelog(done) отработает. Затем он последовательно помещает эти колбеки в основной стек и вызывает. Таски\микротаскинужно почитать, если интересна эта тема. Так на словах сложно обьяснить, довольно пространственная тема
а я понял. процесс завершается если нет задач в тасках и микротасках. верно?
я хз как в ноде завершается процесс )
Неа, не знаю. видел на мдн или ещё где, но не читал и не использовал, хз что такое
В случае браузера, если нет тасок\микротасок\стек пустой\репайнтинг не нужен и тд - то евент луп просто крутится вхолостую и проверяет - не произошла ли какая оказия. Вобщем, таски\микротаски\евент луп из семи залуп - есть крутые видосы на ютубе на английском которые нагляднее чем текст описывают это действо.
Обсуждают сегодня