生成器是由生成器函数( generator function )运行后获得的,是可迭代的。ajax
function* gen() { yield 'a'; yield 'b'; yield 'c'; } let g = gen(); // "Generator { }"
生成器有一个很大的特色,它能够暂停内部代码运行,返回一个值给外部函数。(暂停后不会阻止其余代码运行)当外部调用其 next 方法后,会继续执行剩下的代码,并接受外部传来的一个参数。这个实现主要依赖于关键字 yield 。dom
yield 关键字使生成器函数执行暂停, yield 关键字后面的表达式的值返回给生成器的调用者。它能够被认为是一个基于生成器的版本的 return 关键字。
function* g(){ var a = yield 2; console.log(a); } var it = g(); // 返回一个可迭代的生成器对象 console.log(it.next()); // 执行生成器函数内部代码,第一次返回 {done: false, value: 2} it.next(3); // 继续执行生成器函数内部代码,同时向生成器传递参数3,最后返回 {done: true, value: undefined}
一个简单的计数器异步
function* count(){ var n = 1; while(true){ yield n++; } } var it = count(); it.next(); // 1 it.next(); // 2 it.next(); // 3
之前处理异步 ajax 请求结果,通常采用传递回调函数的方式。一旦遇到多层回调嵌套,代码的可读性会下降,而且调试起来也不方便。有了生成器以后,咱们就能够用同步的方式写异步的代码。这听上去很是的有意思。咱们的代码将会是这样的async
function foo(){ var result = asyncFun(); // asyncFun 是异步函数,result 是异步返回的结果 console.log(result); }
固然,上面的代码并不能获得正确的结果,它只是一个设想。咱们正打算用生成器来实现,并且这是可行的。想一想生成器有哪些特色:函数
这就足够了。有了生成器函数,如今咱们从新来设计代码:设计
function* foo(){ // 这里遇到了异步方法,必须停下来。 // 等待异步方法执行完毕,并返回结果,继续运行代码。固然,同步 ajax 不能算,它不是异步 // 输出结果 }
静下来想想有哪些关键字,与暂停、继续有关。停下来...继续...停下来...继续...停下来...继续...Don't...Stop...Don't...Stop...Don't...Stop......这两个词就是 yield、next.调试
function *foo(){ var result = yield asyncFun(next); console.log(result); }
当代码遇到 yield 会暂停,这个时候 asyncFun 函数是不会暂停的,会执行,等执行完毕,再调用生成器的 next 方法,并将返回结果做为参数传给 next。因为在生成器函数内部咱们拿不到 next,必须借助于全局变量来传递 next。code
var next, gn; function asyncFun(next){ // 模拟异步请求 setTimeout(function(){ // 返回一个随机数 next(Math.random()) }, 1000) } function* foo(){ var result = yield asyncFun(next); console.log(result); } gn = foo(); next = gn.next.bind(gn); next(); // 打印随机数
这样写,运行看上去有些繁重。能够写一个包装函数运行含有异步代码的生成器函数。对象
function asyncFun(next){ // 模拟异步请求 setTimeout(function(){ // 返回一个随机数 next(Math.random()) }, 1000) } function* foo(){ var result = yield function(next){asyncFun(next)}; console.log(result); } function wrapFun (gFn){ var gn = foo(), next = gn.next.bind(gn); next().value(next); } wrapFun(foo);
演示地址get
不过,自从出了 Promise 和 await 以后,更多的是用这个组合,其使用也更简单,范围也更广。