ESM模块执行是同步阻塞式深度优先后序遍历,不进入微任务队列;其evaluate阶段会完全阻塞主线程,导致UI卡顿、掉帧甚至无响应,真正入微任务队列的是模块内显式创建的Promise.then等回调。
ESM 模块本身不直接进入微任务队列执行,它的解析、链接和执行是浏览器加载阶段的同步拓扑过程,与 Promise.then、queueMicrotask 等典型的微任务无直接调度关系。真正影响 UI 响应性的,是 ESM 执行阶段的阻塞行为及其与渲染主线程的交互方式。
ESM 的 evaluate 阶段(即运行模块顶层代码)是同步、阻塞式、深度优先后序遍历的过程。它发生在 HTML 解析暂停期间(对 script type="module"),或在动态 import() 的 Promise resolve 后立即同步执行——此时它会插入到当前调用栈中,而非排队进微任务队列。
这意味着:
ESM 模块体里写的代码,比如:
Promise.resolve().then(() => { /* ... */ })queueMicrotask(() => { /* ... */ })这些才会被推入微任务队列,在当前宏任务(如 ESM evaluate、事件回调)结束后立即执行。但注意:这些微任务的触发时机,仍受限于 ESM 执行是否已完成——如果模块 A 导入了耗时模块 B,那么 B 的 evaluate 必须先完成,A 中定义的微任务才可能被注册和执行。
ESM 的同步执行特性,使它天然成为 UI 卡顿的潜在源头,尤其在以下场景:
缓解策略包括:
import() 拆分非首屏逻辑,把 evaluate 推迟到用户交互后(如点击触发),避免阻塞初始渲染;setTimeout(..., 0) / requestIdleCallback 主动让出主线程;本质上,ESM 的执行优先级不是“比微任务高或低”,而是“在微任务之前就已占据主线程”。理解这一点,才能避开把模块拆分误当作异步优化的常见误区。