Generator

生成器是es6原生提供的异步编程方案,其语法行为和传统函数完全不同,阮大的《ECMAScript 6 入门》一书中对生成器有比较详尽的介绍,还有一些其他的文章可以参考,比如:

  • 《ECMAScript 6 入门:generator》

  • 深入浅出ES6(三):生成器 Generators

  • 深入浅出ES6(十一):生成器 Generators,续篇

本文主要是通过一些代码示例来记录和总结生成器的用法。

yield 和 next

yieldnext在生成器中扮演着非常重要的角色,前者是一个操作符,后者是生成器上的一个函数。

他们具有以下特性:

  1. 需要调用generator的next函数,生成器中的语句才开始执行;

  2. next函数在生成器之外调用,意味着可以在生成器之外控制其内部操作的执行过程;

  3. 当生成器执行到yield操作符就立即执行yield之后的语句并暂停,不敢妄言内部原理,姑且感性地比作savepoint;

  4. 当再次调用生成器的next函数时,生成器从上次发生yield'savepoint'继续执行,直到再次遇到yield,或者遇到是return或者throw生成器就退出;

  5. next的返回值是一个形如{done:false, value:x}的对象,每次调用next都会使生成器继续执行,对于next的返回值有如下规律:

    • 如果再次遇到yieldnext返回值中的value属性是紧接在这条yield之后的语句执行之后的返回值;

    • 如果遇到的是return,那么返回对象done=truevalue则是return的返回值;

    • 其他情况下,返回对象{done:false, value:undefined};

  6. next的输入参数在上一次发生yield的地方返回,所以第一次调用next传入的参数是“然并卵”,next是在生成器之外调用的,所以这个机制使得我们有能力控制生成器内部的行为。

以上说了很多,先看一个用生成器实现的一个无限斐波那契数列,可以无限的调用next函数,他永远不会新视觉返回done=true


  1. const f = function* fibonacci() {
  2. let [a, b] = [0, 1];
  3. for (;;) {
  4. yield a;
  5. [a, b] = [b, a + b];
  6. }
  7. }();
  8. //执行三次,得到三个对象,其value值分别是0,1,1
  9. for (let i of Array(3).keys()) {
  10. console.log(f.next());
  11. }

接下来通过一段代码看看next和yield在传值和返回值上的情况,如下:


  1. const iter = function* gen() {
  2. console.log(`yield ${(yield 'a' + 0)}`);
  3. console.log(`yield ${(yield 'b' + 1)}`);
  4. return 'c' + 2;
  5. }();
  6. console.log(`next:${iter.next(0).value}`); //输出 next:a0
  7. console.log(`next:${iter.next(1).value}`); //输出 yield 1 next:b1
  8. console.log(`next:${iter.next(2).value}`); //输出 yield 2 next:c2

对以上代码的输出分析如下:

  1. 第一个next触发生成器执行到第一个yield,并立即执行'a' + 0 = 'a0'a0作为这次next的返回值;

  2. 第二个带参数为1next触发生成器继续执行,此时第一个yield才返回1,然后执行到第二个yield并立即立即这条yield后面的'b' + 1 = 'b1'b1作为这次next的返回;

  3. 第三个next执行以此类推……

异步编程方案

在同步编程模型中,每个函数总是有序依次地执行,一般上一个函数执行的结果往往是下一个函数的入参,那么在javascript中如何让下一个异步操作等待上一个异步执行得到结果之后再执行呢?

我们现在已经有了生成器并且知道next可以触发生成器执行到yield操作处,而且生成器会在遇到yield时立即执行后面的语句并暂停,那么如果yield后面是一个异步操作,而异步操作获取到结果之后再调用next不就实现了等待新视觉影院的效果么?


  1. function asyncfuc(v) {
  2. setTimeout(function() {
  3. let r = v + 20;
  4. console.log(r);
  5. g.next(r); //把异步函数执行得到的结果传出并触发下一个yield
  6. }, 500);
  7. }
  8. let g = function* gen() {
  9. let v1 = yield asyncfuc(0);
  10. let v2 = yield asyncfuc(v1); //上一个异步调用的结果作为下一个异步调用的入参
  11. return v2;
  12. }();
  13. g.next();

异步操作执行链

有了前文的基础我们可以实现一个用来执行多个异步操作的函数,定义一个run(...functions)方法依次执行传入的函数,如下:


  1. //这个方法用来模拟一个异步调用
  2. function delay(time, callback) {
  3. setTimeout(function () {
  4. callback(`slept for ${time}`);
  5. }, time);
  6. }
  7. function run(...functions) {
  8. //构造一个生成器循环执行传入的方法
  9. var generator = function* sync(resume, functions) {
  10. let result;
  11. for (var func of functions) {
  12. result = yield func(result, resume); //前一个方法执行的结果作为下一个方法的入参
  13. }
  14. return result;
  15. }(resume, functions);
  16. //提供一个方法用于推进生成器执行。
  17. function resume(callbackValue) {
  18. generator.next(callbackValue);
  19. }
  20. generator.next(); //触发生成器立即执行第一个方法
  21. }
  22. //模拟异步方法调用, 斐波那契数列
  23. function d(result, resume) {
  24. delay(1000, (msg) => {
  25. let value = result;
  26. if (value) {
  27. [value.a, value.b] = [value.b, value.a + value.b];
  28. } else {
  29. value = { a: 0, b: 1 };
  30. }
  31. console.log(value.a);
  32. resume(value);
  33. });
  34. return result;
  35. }
  36. run(d, d, d); //顺序执行异步方法

以上实现有个值得注意的地方是通过一个叫做的resume的函数来推进生成器,它要求业务处理函数必须带上一个resume并在获得异步结果之后把结果作为参数并执行resume(value)(如同上面的d(result, resume)函数),由于resume不是一个标准的api,因此这个方案具有比较大的侵入性。

ES6 异步编程之一:Generator的更多相关文章

  1. 转: ES6异步编程:Generator 函数的含义与用法

    转: ES6异步编程:Generator 函数的含义与用法 异步编程对 JavaScript 语言太重要.JavaScript 只有一根线程,如果没有异步编程,根本没法用,非卡死不可. 以前,异步编程 ...

  2. 转: ES6异步编程: co函数库的含义与用法

    转: ES6异步编程: co函数库的含义与用法 co 函数库是著名程序员 TJ Holowaychuk 于2013年6月发布的一个小工具,用于 Generator 函数的自动执行. 比如,有一个 Ge ...

  3. 转: ES6异步编程:Thunk函数的含义与用法

    转: ES6异步编程:Thunk函数的含义与用法 参数的求值策略 Thunk函数早在上个世纪60年代就诞生了. 那时,编程语言刚刚起步,计算机学家还在研究,编译器怎么写比较好.一个争论的焦点是&quo ...

  4. 深入解析js异步编程利器Generator

    我们在编写Nodejs程序时,经常会用到回调函数,在一个操作执行完成之后对返回的数据进行处理,我简单的理解它为异步编程. 如果操作很多,那么回调的嵌套就会必不可少,那么如果操作非常多,那么回调的嵌套就 ...

  5. JavaScript异步编程:Generator与Async

    从Promise开始,JavaScript就在引入新功能,来帮助更简单的方法来处理异步编程,帮助我们远离回调地狱. Promise是下边要讲的Generator/yield与async/await的基 ...

  6. es6异步编程 Promise 讲解 --------各个优点缺点总结

    //引入模块 let fs=require('fs'); //异步读文件方法,但是同步执行 function read(url) { //new Promise 需要传入一个executor 执行器 ...

  7. es6异步编程

    https://blog.csdn.net/tcy83/article/details/80274772 等一系列文章

  8. es6 generator函数的异步编程

    es6 generator函数,我们都知道asycn和await是generator函数的语法糖,那么genertaor怎么样才能实现asycn和await的功能呢? 1.thunk函数    将函数 ...

  9. ES6/7 异步编程学习笔记

    前言 在ES6的异步函数出现之前,Js实现异步编程只有settimeout.事件监听.回调函数等几种方法 settTmeout 这种方法常用于定时器与动画的功能,因为其本质上其实是浏览器的WebAPI ...

随机推荐

  1. bootstrap中container和container-fluid的区别与用法

    对bootstrap框架有一定了解的朋友都知道,一般页面布局中的开头会使用到container或container-fluid类,那么它们有什么区别呢?不急!下面为您讲解. 我们先来看看官方对这两个类 ...

  2. 【TensorFlow入门完全指南】模型篇·线性回归模型

    首先呢,进行import,对于日常写代码来说,第二行经常写成:import numpy as np,这样会更加简洁.第三行import用于绘图. 定义了学习率.迭代数epoch,以及展示的学习步骤,三 ...

  3. python基础教程总结15——5 虚拟茶话会

    聊天服务器: 服务器能接受来自不同用户的多个连接: 允许用户同时(并行)操作: 能解释命令,例如,say或者logout: 容易拓展 套接字和端口: 套接字是一种使用标准UNIX文件描述符(file ...

  4. python基础教程总结14——测试

    1. 先测试,后编码 对程序的各个部分建立测试也是非常重要的(这也称为单元测试).测试驱动编程:Test-driven programming 1)精确的需求说明: 程序设计的理念是以编写测试程序开始 ...

  5. 字符编码:Unicode和UTF-8的关系

    今天中午,我突然想搞清楚Unicode和UTF-8之间的关系,于是就开始在网上查资料. 结果,这个问题比我想象的复杂,从午饭后一直看到晚上9点,才算初步搞清楚. 下面就是我的笔记,主要用来整理自己的思 ...

  6. 天坑之mysql乱码问题以及mysql重启出现1067的错误解决

    相信很多小伙伴都遇到过数据库中文乱码问题,很头疼,明明Navicat上的编码格式都是utf-8是一样的啊? 为什么还是乱码? 原因是Navicat上的数据库编码格式并不是真正的编码格式 ,所以明白了吗 ...

  7. 读取Exchange的用户未读邮件数的几种方法

    [http://www.cnblogs.com/nbpowerboy/p/3539422.html] 可以使用ExchangeServiceBinding获取邮件,他相当于outlook, 来获取服务 ...

  8. C#数组添加元素

    一.向数组添加元素 在C#中,只能在动态数组ArrayList类中向数组添加元素.因为动态数组是一个可以改变数组长度和元素个数的数据类型. 示例: using System;using System. ...

  9. The expected type was 'System.Int64' but the actual value was null.”

    System.InvalidOperationException:“An exception occurred while reading a database value for property ...

  10. 【wqs二分】HHHOJ#15. 赤

    这个wqs二分并不熟练…… 题目描述 #15. 赤 题目分析 两维都用wqs二分,其他没有什么特殊之处. 重点在于,wqs二分还原最优解的时候,增量是强制给的k. #include<bits/s ...