成为 JavaScript 生成器的专家,掌握 JavaScript Generators:5 个实用案例

点击上方 前端Q,关注公众号

回复加群,加入前端Q技术交流群

它们不仅仅是一个花哨的功能,我们将探索它们的多种强大用例,包括创建引人入胜的动画、通过互联网流式传输视频、节省内存等。

如果你从未听说过它们,可能会错过很多有趣的东西。

什么是 Generators?

Generators 是一种神奇的函数,它们可以在任意时刻暂停和恢复执行,而不是连续执行。

使用星号 * 标记函数为生成器,yield.next() 调用时按需生成值,直到生成器完成为止。

e3a3c04c0d323b13f9f53c5aed1df575.png

这就像一个物理发电机,它不是一次性生成所有的电,而是随着时间推移逐渐生成。

5adf9593935f2d7a70a5b97f24c2a43d.gif

你可以使用 for..of 循环来替代直接调用 .next(),这对于生成大量数据的生成器非常有用:

function* numberGenerator() {
  let num = 1;
  while (true) {
    yield num++;
  }
}

for (let value of numberGenerator()) {
  if (value > 5) break;
  console.log(value);
}
// 输出:1, 2, 3, 4, 5

惰性求值

“只在必要时计算”。

与 JavaScript 中的常规函数不同,常规函数会完全执行并返回结果。

假设你想要一个数字序列,但不确定需要多少个。生成器可以帮助你:

function* sequenceGenerator() {
  let num = 1;
  while (true) {
    yield num++;
  }
}

const generator = sequenceGenerator();
console.log(generator.next().value); // 1
console.log(generator.next().value); // 2
// 你可以根据需要继续获取下一个值

使用生成器,你只在需要时获取下一个数字。

更好的内存利用

生成器不会在内存中保存所有结果,而是动态生成。

想象一下,你需要一个包含一百万个数字的序列。使用常规函数,你需要将这些数字存储在一个数组中,消耗大量内存。

生成器效率更高:

function* bigSequence() {
  let i = 0;
  while (i < 1000000) {
    yield i++;
  }
}

const generator = bigSequence();
console.log(generator.next().value); // 0
console.log(generator.next().value); // 1
// 继续按需生成下一个值,而不是一次性生成所有值

处理异步任务

你知道吗?Babel 会将 async/await 转换为生成器,以便在不支持原生 async/await 的 JavaScript 版本中使用。

Babel 将以下代码:

7ce14c726282f136693715c3b3dbec7d.png

转换为:

9fdcb43e3c11dbced9b22da8b80551cf.png

打字动画

打字动画可以吸引用户的注意力,使你的网站更加美观。

它们通过模拟人类的打字行为,为网站增添个性和特色,从而创造更具人性化的体验,并建立独特的品牌形象。

使用递归和 setTimeout() 可能是实现的一个不错的方式:

function typeWriter(text, i = 0) {
  if (i < text.length) {
    document.body.innerHTML += text.charAt(i);
    setTimeout(() => typeWriter(text, i + 1), 100);
  }
}

typeWriter('Hello, World!');

但是,在这种情况下,生成器更能大显身手:

function* typeWriterGenerator(text) {
  let i = 0;
  while (i < text.length) {
    yield text.charAt(i++);
  }
}

const generator = typeWriterGenerator('Hello, World!');
const interval = setInterval(() => {
  const next = generator.next();
  if (next.done) {
    clearInterval(interval);
  } else {
    document.body.innerHTML += next.value;
  }
}, 100);

因为我们可以随时生成值,所以可以使用 setInterval() 以时间间隔生成字符。

异步处理

注意:这与前面提到的 async/await 的基础不同。这里我们谈论的是异步生成器。

0b119b4deb8b91c45c62e53287e78a28.png

我们可以这样使用异步生成器:

480b07344c91cae296ffc7f8e311c0f8.png

这个工具在 Web 应用中以结构化、可读的方式流式传输数据非常强大——看看这个为类似 YouTube 的视频共享应用缓冲和流式传输数据的函数:

8bbe270795925a8b3d505dacd929d99d.png

要消费这个异步生成器,我们使用 for await..of 循环:

d641fee3bea0f5aebe9a09790f21a485.png

redux-saga

redux-saga 是一个用于管理应用程序中副作用的库,拥有超过 100 万的每周下载量。

2f3367816db34a9ff08db83798542f06.png

生成器在这个库中扮演了重要角色,处理 Redux 操作以简化测试和错误处理。

看看这个简单的 saga:

0ae4dc0b66dd9a26334f62311ad3fc1e.png

f30dbb57c3f86eba84fbe48efb77ca88.png

每当 USER_FETCH_REQUESTED 操作被派发时,redux-saga 运行生成器,它依次调用 fetchData() 以执行异步网络请求。

关于 return 的说明

在生成器函数中返回值时会发生什么?让我们看看:

function* footballPlayers() {
  yield 'Messi';
  yield 'Ronaldo';
  return 'Neymar';
}

for (const player of footballPlayers()) {
  console.log(player);
}
// 输出:Messi, Ronaldo

为什么 Neymar 不包含在生成的值中?

我们使用 .next() 来看看 done 属性是否有关系:

const players = footballPlayers();
console.log(players.next()); // { value: 'Messi', done: false }
console.log(players.next()); // { value: 'Ronaldo', done: false }
console.log(players.next()); // { value: 'Neymar', done: true }

结果显示,return 不是生成的值,因此 for..of 不会处理它。

你还记得我们的第一个例子吗?

function* numberGenerator() {
  let num = 1;
  while (num <= 3) {
    yield num++;
  }
  return 'done';
}

const generator = numberGenerator();
for (const value of generator) {
  console.log(value);
}
// 输出:1, 2, 3

你可以看到,生成器只生成值,直到 donetrue

因此,return 会完成生成器并终止函数(就像其他函数一样)。

最后总结

JavaScript 生成器为控制流、内存效率和异步处理提供了强大的解决方案。它们通过动态动画、流式数据和管理副作用增强了 Web 开发。

让我们拥抱生成器的多功能性,编写优雅高效的 JavaScript 代码。

53354bcb856ed6935f2ee44ff08fd997.png

往期推荐

基于web-see的前端监控方案实现

52312c983122b0be51a3db7d2c249998.png

用了这些 Vite 配置技巧,同事都以为我开挂了!(5000 字干货,建议收藏)

7e5cca467c53cfe049a83e455b88adb7.png

放弃 React,微软 Edge 团队改用 Web 组件减少对 JavaScript 的依赖

1c4f7f87d2ca4274997851925e39cc64.png


最后

  • 欢迎加我微信,拉你进技术群,长期交流学习...

  • 欢迎关注「前端Q」,认真学前端,做个专业的技术人...

c2f7a2579d8ec97bad17c6cff2b2254d.jpeg

a39c3affa969b41fc474591eaa4132a9.png

点个在看支持我吧

df02bbf86f8c177f795e4c6467c19fb1.gif

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值