[4-3] 异步编程与事件循环 · 终极异步解决方案 (Generator & Async/Await)

所属板块:4. 异步编程与事件循环

记录日期:2026-03-xx

更新:遇到 async/await 输出题或并发控制题时补充

1. Generator 函数(过渡期方案)

Generator 是 ES6 引入的"可暂停/恢复执行"的函数,用 function* 定义,yield 暂停。

js 复制代码
function* gen() {
  yield 1;
  yield 2;
  return 3;
}

const g = gen();
console.log(g.next());   // { value: 1, done: false }
console.log(g.next());   // { value: 2, done: false }
console.log(g.next());   // { value: 3, done: true }

它本身不能直接处理异步,但结合执行器(如 co 库)可以实现异步流程控制,为 async/await 铺路。

2. async/await 的本质(Generator + Promise 语法糖)

async/await 是 Generator + Promise 的语法糖:

  • async 函数的返回值一定是一个 Promise
  • await 后面如果不是 Promise,会被自动 Promise.resolve() 包装
  • await 下方的代码会被包装成 .then() 回调,推入微任务队列(结合 [4-1] 事件循环)
js 复制代码
async function test() {
  console.log("1");
  await Promise.resolve("await");
  console.log("2");
}

test();
console.log("3");

输出:1 → 3 → 2(await 转为微任务)

3. await 在 Event Loop 中的真实表现(极其重要)

  • await 右侧表达式同步执行
  • await 下方的所有代码被包装成微任务
  • 如果 await 的是已 resolved 的 Promise,仍会推入微任务队列(不会立即执行)

经典对比:

js 复制代码
async function asyncFunc() {
  console.log("async start");
  await Promise.resolve();
  console.log("async end");
}

asyncFunc();
console.log("global");

输出:async start → global → async end

4. 实战场景与错误处理

推荐写法(try...catch)

js 复制代码
async function fetchData() {
  try {
    const res = await fetch('/api');
    const data = await res.json();
    return data;
  } catch (err) {
    console.error("请求失败", err);
  }
}

并发优化误区(最容易踩坑)

js 复制代码
// 错误写法:串行执行
for (const url of urls) {
  const data = await fetch(url);   // 一个接一个
}

// 正确写法:先并发启动,再 await
const promises = urls.map(url => fetch(url));
const results = await Promise.all(promises);

5. 小结 & 第四板块完结

  • Generator 是"可暂停函数",async/await 是其语法糖 + Promise 结合
  • await 本质是微任务调度(必须结合 [4-1] 事件循环理解)
  • [4-1] 事件循环 + [4-2] Promise + 本文 async/await,共同构成了 JS 异步编程的完整体系

第四板块(异步编程与事件循环)到此全部结束。

这是 JS 单线程下最核心的"动态调度机制",掌握后刷输出题和写并发代码会非常顺手。

返回总目录:戳这里

相关推荐
不甜情歌2 小时前
JS 拷贝:浅拷贝 / 深拷贝原理 + 常用方法
前端·javascript
Roselind_Yi2 小时前
技术拆解:《从音频到动效:我是如何用 Web Audio API 拆解音乐的?》
前端·javascript·人工智能·音视频·语音识别·实时音视频·audiolm
我叫黑大帅2 小时前
Js常用数组处理
前端·javascript·面试
我叫黑大帅2 小时前
Js常用的字符串处理
前端·javascript·面试
读忆2 小时前
在前端开发中使用组件后, 若是出了bug, 应该如何排查, 怎么排查, 解决方式是什么?
前端·javascript·vue.js·bug
Aevget2 小时前
界面控件DevExpress JS & ASP.NET Core v25.2新版亮点 - AI驱动扩展功能升级
javascript·asp.net·界面控件·devexpress·ui开发
SuperEugene2 小时前
Vue3 + Vue Router + Pinia 路由守卫规范:beforeEach 应做 / 不应做,避死循环、防重复请求|状态管理与路由规范篇
开发语言·前端·javascript·vue.js·前端框架
东东__net2 小时前
js逆向与谷歌加密库
开发语言·前端·javascript
m0_502724952 小时前
腾讯地图tlbs-multi-marker动态更新marker图标
前端·javascript·vue.js·地图