js 同步异步,宏任务微任务的关系

一、先分清:同步 vs 异步

1. 同步(Sync)

  • 代码按顺序执行
  • 必须等上一行执行完,才能执行下一行
  • 会阻塞代码
  • 例子:for 循环、普通赋值、console.log、计算
js 复制代码
console.log(1)
console.log(2)
console.log(3)
// 输出:1 2 3

2. 异步(Async)

  • 不会立刻执行,先放到"任务队列"
  • 所有同步代码执行完,才会执行异步
  • 不阻塞代码
  • 例子:定时器、ajax/fetch、Promise、事件
js 复制代码
console.log(1)
setTimeout(() => console.log(2), 0)
console.log(3)
// 输出:1 3 2

二、核心重点:宏任务(Macrotask) vs 微任务(Microtask)

JS 异步任务分两类:宏任务微任务

执行规则(死记硬背)

  1. 先执行所有同步代码
  2. 再执行所有微任务
  3. 最后执行宏任务

口诀:
同步 → 微任务 → 宏任务


三、哪些是宏任务?哪些是微任务?

宏任务(慢)

  • setTimeout
  • setInterval
  • Ajax/fetch
  • 事件监听(click、resize)
  • requestAnimationFrame
  • UI 渲染

微任务(快)

  • Promise.then / catch / finally
  • async/await(本质就是 Promise)
  • queueMicrotask
  • Vue nextTick
  • Vue 的 DOM 更新、watch 回调

四、最经典的执行顺序题(必看)

js 复制代码
console.log('1 同步')

setTimeout(() => {
  console.log('2 宏任务')
}, 0)

Promise.resolve().then(() => {
  console.log('3 微任务')
})

console.log('4 同步')

输出结果:

复制代码
1 同步
4 同步
3 微任务
2 宏任务

五、完整执行流程(底层原理)

  1. 执行同步代码
  2. 同步执行完 → 清空所有微任务
  3. 微任务清空 → 执行一个宏任务
  4. 宏任务执行完 → 又清空所有微任务
  5. 循环往复...

六、和 Vue 的关系(超级重要)

  • Vue 异步更新 DOM
  • nextTick
  • watch 的 flush

全部都是 微任务!

所以 Vue 事件循环顺序:
同步代码 → Vue DOM 更新(微任务)→ nextTick(微任务)→ 宏任务(setTimeout)


七、超级总结(必背)

  1. 同步先跑,异步后跑
  2. 异步分两种:微任务 > 宏任务
  3. 执行顺序:同步 → 微任务 → 宏任务
  4. Promise/await/nextTick 是微任务
  5. setTimeout/ajax 是宏任务

一句话终极记忆

同步最先,微任务中间,宏任务最后!

需要我给你出 3 道前端面试必考的执行顺序题 让你练手吗?一做就彻底吃透。

相关推荐
weedsfly5 分钟前
前端必知必会:从 IIFE 到 ESM,模块化到底在解决什么?
前端·javascript
渣波8 分钟前
拒绝 SQL 焦虑!手把手带你用 NestJS + Prisma + DTO 写出“防弹”级后端代码
javascript·数据库·后端
槑有老呆12 分钟前
每次跟大模型聊天,都是一次「失忆」的 HTTP 请求
javascript
sarasuki14 分钟前
彻底搞懂JS闭包:从作用域链、形成条件到优缺点
javascript
糖拌西瓜皮16 分钟前
TypeScript 进阶:泛型、条件类型、类型守卫与装饰器
javascript·node.js
swipe14 小时前
从 0 到 1 实现大文件上传:分片、秒传、断点续传、暂停、重试与服务端合并
前端·javascript·面试
kyriewen15 小时前
AI 生成的代码能跑就行?这 5 个坑迟早炸
前端·javascript·ai编程
kisshyshy15 小时前
🍦 雪糕、食堂、火车厢:三幅漫画吃透栈、队列与链表
javascript·算法
胡志辉16 小时前
从v8源码和react深入浅出理解 JavaScript 作用域链与闭包
前端·javascript
Bolt16 小时前
TypeScript 7.0 来了:当 tsc 用 Go 重写之后
javascript·typescript·go