Generator函数跟普通函数的写法有很是大的区别:javascript
一是,function关键字与函数名之间有一个星号;
二是,函数体内部使用yield语句,定义不一样的内部状态(yield在英语里的意思就是“产出”)。html
最简单的Generator函数以下:java
function* g() { yield 'a'; yield 'b'; yield 'c'; return 'ending'; } g(); // 返回一个对象
g函数呢,有四个阶段,分别是'a','b','c','ending'。jquery
g()
并不会执行g函数,返回的也不是函数运行结果,而是一个指向内部状态的指针对象,也就是迭代器对象(Iterator Object)。ajax
先看以下代码:编程
function* g() { yield 'a'; yield 'b'; yield 'c'; return 'ending'; } var gen = g(); gen.next(); // 返回Object {value: "a", done: false}
gen.next()
返回一个很是很是简单的对象{value: "a", done: false}
,'a'就是g函数执行到第一个yield语句以后获得的值,false表示g函数尚未执行完,只是在这暂停。数组
若是再写一行代码,仍是gen.next();
,这时候返回的就是{value: "b", done: false}
,说明g函数运行到了第二个yield语句,返回的是该yield语句的返回值'b'。返回以后依然是暂停。promise
再写一行gen.next();
返回{value: "c", done: false}
,再写一行gen.next();
,返回{value: "ending", done: true}
,这样,整个g函数就运行完毕了。安全
提问:若是再写一行gen.next();
呢?
答:返回{value: undefined, done: true}
,这样没意义。服务器
提问:若是g函数没有return语句呢?
答:那么第三次.next()
以后就返回{value: undefined, done: true}
,这个第三次的next()
惟一意义就是证实g函数所有执行完了。
提问:若是g函数的return语句后面依然有yield呢?
答:js的老规定:return语句标志着该函数全部有效语句结束,return下方还有多少语句都是无效,白写。
提问:若是g函数没有yield和return语句呢?
答:第一次调用next就返回{value: undefined, done: true}
,以后也是{value: undefined, done: true}
。
提问:若是只有return语句呢?
答:第一次调用就返回{value: xxx, done: true}
,其中xxx
是return语句的返回值。以后永远是{value: undefined, done: true}
。
提问:下面代码会有什么结果?
function* g() { var o = 1; yield o++; yield o++; yield o++; } var gen = g(); console.log(gen.next()); // 1 var xxx = g(); console.log(gen.next()); // 2 console.log(xxx.next()); // 1 console.log(gen.next()); // 3
答:见上面注释。每一个迭代器之间互不干扰,做用域独立。
继续提问:若是第二个yield o++;
改为yield;
会怎样?
答:那么指针指向这个yield的时候,返回{value: undefined, done: false}
。
继续提问:若是第二个yield o++;
改为o++;yield;
会怎样?
答:那么指针指向这个yield的时候,返回{value: undefined, done: false}
,由于返回的永远是yield后面的那个表达式的值。
因此如今能够看出,每次调用next方法,内部指针就从函数头部或上一次停下来的地方开始执行,直到遇到下一个yield语句(或return语句)为止。换言之,Generator函数是分段执行的,yield语句是暂停执行的标记,而next方法能够恢复执行。
总之,每调用一次Generator函数,就返回一个迭代器对象,表明Generator函数的内部指针。之后,每次调用迭代器对象的next方法,就会返回一个有着value和done两个属性的对象。value属性表示当前的内部状态的值,是yield语句后面那个表达式的值;done属性是一个布尔值,表示是否遍历结束。
因此能够看出,Generator 函数的特色就是:
一、分段执行,能够暂停
二、能够控制阶段和每一个阶段的返回值
三、能够知道是否执行到结尾
迭代器对象的next方法的运行逻辑以下。
(1)遇到yield语句,就暂停执行后面的操做,并将紧跟在yield后面的那个表达式的值,做为返回的对象的value属性值。
(2)下一次调用next方法时,再继续往下执行,直到遇到下一个yield语句。
(3)若是没有再遇到新的yield语句,就一直运行到函数结束,直到return语句为止,并将return语句后面的表达式的值,做为返回的对象的value属性值。
(4)若是该函数没有return语句,则返回的对象的value属性值为undefined。
yield语句与return语句既有类似之处,也有区别。
类似之处在于,都能返回紧跟在语句后面的那个表达式的值。
区别在于每次遇到yield,函数暂停执行,下一次再从该位置继续向后执行,而return语句不具有位置记忆的功能。一个函数里面,只能执行一次(或者说一个)return语句,可是能够执行屡次(或者说多个)yield语句。正常函数只能返回一个值,由于只能执行一次return;Generator函数能够返回一系列的值,由于能够有任意多个yield。从另外一个角度看,也能够说Generator生成了一系列的值,这也就是它的名称的来历(在英语中,generator这个词是“生成器”的意思)。
注意:yield语句只能用于function*
的做用域,若是function*
的内部还定义了其余的普通函数,则函数内部不容许使用yield语句。
注意:yield语句若是参与运算,必须用括号括起来。
console.log(3 + yield 4); // 语法错误 console.log(3 + (yield 4)); // 打印7
一句话说,next方法参数的做用,是覆盖掉上一个yield语句的值。
function* g() { var o = 1; var a = yield o++; console.log('a = ' + a); var b = yield o++; } var gen = g(); console.log(gen.next()); console.log('------'); console.log(gen.next(11));
获得:
首先说,console.log(gen.next());
的做用就是输出了{value: 1, done: false}
,注意var a = yield o++;
,因为赋值运算是先计算等号右边,而后赋值给左边,因此目前阶段,只运算了yield o++
,并无赋值。
而后说,console.log(gen.next(11));
的做用,首先是执行gen.next(11)
,获得什么?首先:把第一个yield o++
重置为11,而后,赋值给a,再而后,console.log('a = ' + a);
,打印a = 11
,继续而后,yield o++
,获得2,最后打印出来。
从这咱们看出了端倪:带参数跟不带参数的区别是,带参数的状况,首先第一步就是将上一个yield语句重置为参数值,而后再照常执行剩下的语句。总之,区别就是先有一步先重置值,接下来其余全都同样。
这个功能有很重要的语法意义,经过next方法的参数,就有办法在Generator函数开始运行以后,继续向函数体内部注入值。也就是说,能够在Generator函数运行的不一样阶段,从外部向内部注入不一样的值,从而调整函数行为。
提问:第一个.next()能够有参数么?
答:设这样的参数没任何意义,由于第一个.next()的前面没有yield语句。
for...of循环能够自动遍历Generator函数时生成的Iterator对象,且此时再也不须要调用next方法。for...of循环的基本语法是:
for (let v of foo()) { console.log(v); }
其中foo()
是迭代器对象,能够把它赋值给变量,而后遍历这个变量。
function* foo() { yield 1; yield 2; yield 3; yield 4; yield 5; return 6; } let a = foo(); for (let v of a) { console.log(v); } // 1 2 3 4 5
上面代码使用for...of循环,依次显示5个yield语句的值。这里须要注意,一旦next方法的返回对象的done属性为true,for...of循环就会停止,且不包含该返回对象,因此上面代码的return语句返回的6,不包括在for...of循环之中。
下面是一个利用Generator函数和for...of循环,实现斐波那契数列的例子。
斐波那契数列是什么?它指的是这样一个数列 0, 1, 1, 2, 3, 5, 8, 13, 21, 34, 55, 89, 144........
这个数列前两项是0和1,从第3项开始,每一项都等于前两项之和。
function* fibonacci() { let [prev, curr] = [0, 1]; for (;;) { // 这里请思考:为何这个循环不设定结束条件? [prev, curr] = [curr, prev + curr]; yield curr; } } for (let n of fibonacci()) { if (n > 1000) { break; } console.log(n); }
Generator函数返回的迭代器对象,都有一个throw方法,能够在函数体外抛出错误,而后在Generator函数体内捕获。
既然个人文章是简单理解Generator函数,因此错误捕获直接跳过。
Generator函数返回的迭代器对象,还有一个return方法,能够返回给定的值,而且终结遍历Generator函数。
function* gen() { yield 1; yield 2; yield 3; } var g = gen(); console.log(g.next()); // { value: 1, done: false } console.log(g.return('foo')); // { value: "foo", done: true } console.log(g.next()); // {value: undefined, done: true}
就是说,return的参数值覆盖本次yield语句的返回值,而且提早终结遍历,即便后面还有yield语句也一概无视。
提问:return方法跟next方法的区别都有哪些?
答:
一、return终结遍历,以后的yield语句都失效;next返回本次yield语句的返回值。
二、return没有参数的时候,返回{ value: undefined, done: true }
;next没有参数的时候返回本次yield语句的返回值。
三、return有参数的时候,覆盖本次yield语句的返回值,也就是说,返回{ value: 参数, done: true }
;next有参数的时候,覆盖上次yield语句的返回值,返回值可能跟参数有关(参数参与计算的话),也可能跟参数无关(参数不参与计算)。
若是你打算在Generater函数内部,调用另外一个Generator函数,默认状况下是没有效果的。好比:
function* foo() { yield 'a'; yield 'b'; } function* bar() { yield 'x'; foo(); yield 'y'; } for (let v of bar()){ console.log(v); } // "x" // "y"
可见,并无遍历出'a'和'b'。那么若是想在一个Generator函数里调用另外一个Generator函数,怎么办?用yield*语句。好比:
function* bar() { yield 'x'; yield* foo(); yield 'y'; } // 上个函数等同于 function* bar() { yield 'x'; yield 'a'; yield 'b'; yield 'y'; } // 也等同于 function* bar() { yield 'x'; for (let v of foo()) { yield v; } yield 'y'; } for (let v of bar()){ console.log(v); } // "x" // "a" // "b" // "y"
也就是说,咱们约定被调用的Generator函数为A函数,调用A函数的Generator函数为B函数。yield*
语句的做用,就是遍历一遍A函数的迭代器对象。A函数(没有return语句时)是for...of的一种简写形式,彻底能够用for...of替代yield*
。反之,因为B函数的return语句,不会被yield*
遍历,因此须要用var value = yield* iterator
的形式获取return语句的值。
function *foo() { yield 2; yield 3; return "foo"; } function *bar() { yield 1; var v = yield *foo(); console.log( "v: " + v ); yield 4; } var it = bar(); it.next() // {value: 1, done: false} it.next() // {value: 2, done: false} it.next() // {value: 3, done: false} it.next(); // "v: foo" // {value: 4, done: false} it.next() // {value: undefined, done: true}
上面代码在第四次调用next方法的时候,屏幕上会有输出,这是由于函数foo的return语句,向函数bar提供了返回值。
提问:若是不写*会怎样?
答:yield语句会返回迭代器对象。
提问:若是写两遍yield* foo();
会获得什么?
答:
a b a b
提问:若是yield*语句后面跟着一个数组会怎样?
答:
function* gen(){ yield* ["a", "b", "c"]; } gen().next() // { value:"a", done:false }
这说明,任何数据结构只要有Iterator接口,就能够被yield*遍历。数组有这个接口。
Generator能够暂停函数执行,返回任意表达式的值。这种特色使得Generator有多种应用场景。
Generator是实现状态机的最佳结构。好比,下面的clock函数就是一个常规写法的状态机。
var ticking = true; var clock = function() { if (ticking) console.log('Tick!'); else console.log('Tock!'); ticking = !ticking; }
上面代码的clock函数一共有两种状态(Tick和Tock),每运行一次,就改变一次状态。这个函数若是用Generator实现,就是下面这样。
var clock = function*() { while (true) { console.log('Tick!'); yield; console.log('Tock!'); yield; } };
能够看到,Generator 函数实现的状态机不用设初始变量,不用切换状态,上面的Generator函数实现与ES5实现对比,能够看到少了用来保存状态的外部变量ticking,这样就更简洁,更安全(状态不会被非法篡改)、更符合函数式编程的思想,在写法上也更优雅。Generator之因此能够不用外部变量保存状态,是由于它自己就包含了第一个状态和第二个状态。
下面这个天然段很是重要!很是重要!很是重要!
Generator函数的暂停执行的效果,意味着能够把异步操做写在yield语句里面,等到调用next方法时再日后执行。这实际上等同于不须要写回调函数了,由于异步操做的后续操做能够放在yield语句下面,反正要等到调用next方法时再执行。因此,Generator函数的一个重要实际意义就是用来处理异步操做,改写回调函数。
举个例子,好比我在测试服务器的某目录建了4个文件,分别是'test.html'、'a.html'、'b.html'、'c.html',后三个文件的文件内容跟文件名相同,如今我编辑'test.html'的代码,想要先ajax-get相对网址'a.html',而后再回调里ajax-get相对网址'b.html',而后在回调里ajax-get相对网址'c.html',常规的写法是(用上jQuery):
$.get('a.html',function(dataa) { console.log(dataa); $.get('b.html',function(datab) { console.log(datab); $.get('c.html',function(datac) { console.log(datac); }); }); }); // a.html // b.html // c.html
能够看到,就算用上jquery,也依然是回调地狱的既视感,对不对?那么改为生成器函数写法是:
function request(url) { $.get(url, function(response){ it.next(response); }); } function* ajaxs() { console.log(yield request('a.html')); console.log(yield request('b.html')); console.log(yield request('c.html')); } var it = ajaxs(); it.next(); // a.html // b.html // c.html
能够看到,输出结果也是这样。咱们分析一下:
首先咱们定义了一个普通的request函数,初步分析它的做用是:接受一个url参数,经过异步操做获得response,而后把response做为参数传给it.next(),执行it.next()
。可能你还没看懂,不要紧,继续看:
接着咱们定义了一个叫ajaxs的生成器函数,它的代码挺整齐的。没看懂也没关系,先不说它。
最后是两个语句var it = ajaxs(); it.next();
,这两句最简单,你固然能看懂,就是定义一个叫it的迭代器对象,而后执行it.next();
。
当执行了it.next();
以后,开始遍历ajaxs()对象。ajaxs函数的执行顺序在这必须讲,由于它是异步代码表现改写成同步代码表现的核心关键。记住简单一句话:只有当yield后面跟的函数先执行完,不管执行体里面有多少异步回调,都要等全部回调先执行完,才会执行等号赋值,以及再后面的操做。这也是yield最大的特性。你可能会说,怎么前面那么多文字都从没提过yield竟然这么牛逼呢?由于前面的例子为了最简单化,并无让yield后面跟函数,而是跟了简单值,这并不能体现出生成器函数的优点,由于根本哪也没异步嘛。
还记得我写的《Promises究竟是个啥?》里面关于Promise构造函数的超能力吗?yield的超能力就跟Promise构造函数的超能力差很少:
Promises写法的本质就是把异步写法撸成同步写法。要作这么酷炫这么变态的事情,固然须要Promise构造函数有超能力,它的超能力就是传入Promise构造函数的函数参数会第一优先执行,不管这个函数多么的繁复,有多少层回调,有多少秒的计数器,通通都会最优先执行,也就是说,咱们只要new了一个Promise(),那么Promise构造函数的函数参数就是最高优先级执行,一直到new出一个promise对象实例,后面的代码才会执行。
想象一下,若是yield没有这种超能力,那么,下面a、b、c三行几乎同时执行,谁先得到响应鬼才知道,这就没法保证get a得到响应以后才去get b,get b得到响应以后才get c。
console.log(yield request('a.html')); console.log(yield request('b.html')); console.log(yield request('c.html'));
回到原话题,ajaxs函数执行的第一步是request('a.html')
,这是一个异步函数,但不要紧,JS引擎会耐心等它执行完,它执行的第一步是向a.html发请求,回调执行it.next(response)
,也就是把response传递给it.next()
,这就有趣味了,这个next是第几个next?第二个。由于最初已经执行了一个了。如今有种什么感受?没错,迭代的感受。再复习一下next的参数,.next(response)
意味着什么?意味着覆盖上一个yield语句的返回值。而后,yield request('a.html')
将迭代暂停,然而下一个迭代已经开始了。
最终造成了什么?在每个阶段开始,next(参数)干了两件事,第一件事是用参数覆盖前一个yield语句的值,第二件事是执行本阶段的代码,这样不断迭代下去,最终造成了一个next触发了一串next。这就造成了一个现象:最开始的一个.next()触发了一连串的request函数的执行,不管啥时候我想要执行这一串异步操做,我都只须要两行代码:var it = ajaxs(); it.next();
就够了。够短吧?
妙不妙?
最后一个问题:怎样最快最简单地写出采用 Generator 函数的同步形式的代码?
第1步:将全部异步代码的每一步都封装成一个普通的、能够有参数的函数,好比上面的request函数。你可能问,上面例子为啥三个异步代码却只定义了一个request函数?由于request函数能复用的嘛。若是不能复用的话,请老老实实定义三个普通函数,函数内容就是须要执行的异步代码。
第2步:定义一个生成器函数,把流程写进去,彻底的同步代码的写法。生成器函数能够有参数。
第三步:定义一个变量,赋值为迭代器对象。迭代器对象能够加参数,参数一般将做为流程所需的初始值。
第四步:变量名.next()。不要给这个next()传参数,传了也没用,由于它找不到上一个yield语句。
上面的例子是最简单举例,没有涉及到下一步借用上一步的执行结果的状况,若是想让下一步借用上一步的执行结果的话,其实也简单,好比,我想把a.html的响应内容当作参数,发给b.html,把b.html的响应内容当作参数,发给c.html,也很简单,很少说。
而后咱们再对比一下,Promise写法是怎样:
new Promise(function(resolve) { $.get('a.html',function(dataa) { console.log(dataa); resolve(); }); }).then(function(resolve) { return new Promise(function(resolve) { $.get('b.html',function(datab) { console.log(datab); resolve(); }); }); }).then(function(resolve) { $.get('c.html',function(datac) { console.log(datac); }); });
Promise的写法的优势就是理解起来很简单,每一步中间用then一连就OK。
Promise的写法的缺点就是各类promise实例对象跟一连串的then,代码量大、行数多,满眼的promise、then、resolve看得头晕,并且每个then都是一个独立的做用域,传递参数痛苦。
再举一例,我想在上述每一步异步中间,都间隔3秒。怎么写?
function request(url) { $.get(url, function(response){ it.next(response); }); } function sleep(time) { setTimeout(function() { console.log('I\'m awake.'); it.next(); }, time); } function* ajaxs(ur) { console.log(yield request(ur)); yield sleep(3000); console.log(yield request('b.html')); yield sleep(3000); console.log(yield request('c.html')); } var it = ajaxs('a.html'); it.next();
是否是跟Promise写法的差异更明显了?ajaxs生成器函数里面的代码彻底是同步写法表现。
总之,Generator 函数是比Promise写法更科学的一种写法,实践中应当尽可能使用Generator 函数。