带你用 Javascript 生成器玩转「会暂停」的函数

很多同学一看到 function*yield 就本能地想关文章:看起来又难、又少用。

但其实,生成器(Generator)是 JS 里少见的"会暂停的函数",写好了不仅能让异步逻辑更优雅,还能在处理流式数据(比如大模型回复、SSE 等)时非常顺手。

这篇文章不讲花里胡哨的理论,而是带你一步步搞清楚:

  • 生成器到底是什么?和普通函数有什么本质区别?
  • 怎么用 next()for...of 去"消费"一个生成器?
  • 在 Node.js / 浏览器里,生成器在哪些场景是真的好用?
  • 如何结合 SSE,把大模型这种流式输出封装成一个好用的异步生成器?

看完你可以不爱它,但至少不会再怕它

1. 生成器是什么?一句话版本

用一句话概括:

生成器函数 = 可以中途 yield 暂停、之后再从上次暂停位置继续执行的函数。

它有两点和普通函数完全不一样:

  1. 不会立刻执行:调用生成器函数时,不会马上跑代码,而是返回一个"生成器对象"。
  2. 执行过程可中断 :在函数内部用 yield 抛出一个值的同时,函数会暂停;下次继续从暂停处往后执行。

生成器函数用 function* 定义,最简单的例子长这样:

javascript 复制代码
function* generator() {
    yield 1;
    yield 2;
    yield 3;
}

不管是在 Node.js 还是浏览器中,生成器都已经是非常标准的一部分了。

2. 生成器第一次调用时,到底发生了什么?

重点:调用生成器函数本身,不会执行函数体。

javascript 复制代码
function* generator() {
    yield 1;
    yield 2;
    yield 3;
}

const gen = generator(); // 这里不会执行函数体

这一步返回的 gen 是一个特殊的迭代器对象,只有在你开始"消费"它时,函数体才会真正执行。每一次消费,都通过 next() 完成。

3. 用 next() 消费生成器:一段一段地执行

生成器最常见的使用方式,就是直接调用 next():

javascript 复制代码
const gen = generator();
console.log(gen.next()); // { value: 1, done: false }
console.log(gen.next()); // { value: 2, done: false }
console.log(gen.next()); // { value: 3, done: false }
console.log(gen.next()); // { value: undefined, done: true }

可以看到:

  • value:本次 yield 抛出的值
  • done:生成器是否已经执行完毕
    • false:后面还有东西
    • true:已经结束,再调也是 { value: undefined, done: true }

这里有两个容易忽略的特性:

  • 同一个生成器实例(gen)只能从头到尾走一次。
  • 你可以根据自己的节奏,在任何时间点调用 next(),这就是它"可以暂停"的本质。

4. 用 for...of 优雅遍历生成器

手动 next() 有点繁琐,生成器本身就是可迭代对象,所以可以直接用 for...of 来消费:

javascript 复制代码
const gen = generator();

for (const value of gen) {
    console.log(value);
}
// 1
// 2
// 3

for...of 会在内部帮你处理 next(),直到 donetrue 为止。所以你可以把生成器当成一个自定义规则的"数据流"来看待:你决定它一次 yield 出什么、什么时候结束。

5. 生成器能干嘛?几个典型场景

很多人印象里,生成器只和"高级用法""底层库"绑定在一起,但在 Node.js 和浏览器实际开发中,有几个场景非常适合用它:

  • 控制复杂的异步流程(早期 co 库那一代玩法)
  • 构造自定义迭代器(比如分页数据、树遍历等)
  • 处理流式数据(重点:和大模型 / SSE 这种「一小段一小段不断回来」的数据非常契合)

现在有了 async/await 之后,前两类常见程度下降了一些,但在流式场景里,生成器依然非常好用。

相关推荐
Hilaku10 小时前
我是如何用一行 JS 代码,让你的浏览器内存瞬间崩溃的?
前端·javascript·node.js
五仁火烧10 小时前
npm run build命令详解
前端·vue.js·npm·node.js
前端付豪10 小时前
NodeJs 做了什么 Fundamentals Internals
前端·开源·node.js
局外人LZ13 小时前
libsodium.js:web端与 Node.js 的现代加密工具集,构建前端安全加密体系
前端·javascript·node.js
寂夜了无痕13 小时前
pnpm:快速、节省空间的 Node.js 包管理器
npm·node.js·pnpm
程序员爱钓鱼14 小时前
Node.js 博客系统实战(一):项目需求分析
前端·后端·node.js
Jing_Rainbow1 天前
【Vue-2/Lesson62(2025-12-10)】模块化与 Node.js HTTP 服务器开发详解🧩
前端·vue.js·node.js
TE-茶叶蛋1 天前
NestJS中使用TypeORM
node.js
Drift_Dream1 天前
Node.js 第3课:Express.js框架入门
node.js