时间:2021-05-28
本文实例讲述了ES6新特性三: Generator(生成器)函数。分享给大家供大家参考,具体如下:
1. 简介
① 理解:可以把它理解成一个函数的内部状态的遍历器,每调用一次,函数的内部状态发生一次改变。
② 写法:
function* f() {}③ 作用:就是可以完全控制函数的内部状态的变化,依次遍历这些状态。
④ 运行过程:当调用Generator函数的时候,该函数并不执行,而是返回一个遍历器(可以理解成暂停执行)。通过调用next()开始执行,遇到yield停止执行,返回一个value属性值为当前yield语句的值,done属性为false的对象,循环调用next(),一直执行到return语句(如果没有return语句,就执行到函数结束)。next方法返回的对象的value属性,就是紧跟在return语句后面的表达式的值(如果没有return语句,则value属性的值为undefined),done属性的值true,表示遍历已经结束。
示例:
function* helloWorldGenerator() { yield 'hello'; yield 'world'; return 'ending';}var hw = helloWorldGenerator();//第一次调用该方法不会执行,仅返回一个遍历器。var a = hw.next();while(!a.done){ //当执行到return 时,a.done=true,终止循环 console.log(a.value+','+a.done); a = hw.next();}console.log(a.value+','+a.done);结果:
hello,falseworld,falseending,true2. next() 的参数
① 我们要知道是next()返回一个对象,yield语句本身是没有返回值,或者说总是返回undefined。next方法可以带一个参数,该参数就会被当作上一个yield语句的返回值。
function* f() { for(var i=0; true; i++) { var reset = yield i; console.log(reset); //打印reset,验证yield语句是没有返回值的 if(reset) { i = -1; } }}var g = f();console.log(g.next()) // { value: 0, done: false }console.log(g.next())// { value: 1, done: false }console.log(g.next(true)) // { value: 0, done: false }结果:
{ value: 0, done: false }undefined{ value: 1, done: false }true{ value: 0, done: false }通过next方法的参数,就有办法在Generator函数开始运行之后,继续向函数体内部注入值。也就是说,可以在Generator函数运行的不同阶段,从外部向内部注入不同的值,从而调整函数行为。
②由于next方法的参数表示上一个yield语句的返回值,所以第一次使用next方法时,不能带有参数。V8引擎直接忽略第一次使用next方法时的参数,只有从第二次使用next方法开始,参数才是有效的。
function* foo(x) { var y = 2 * (yield (x + 1)); var z = yield (y / 3); return (x + y + z);}var it = foo(5);console.log(it.next(3))// { value:6, done:false }console.log(it.next(12))// { value:8, done:false }console.log(it.next(13))// { value:42, done:true }3. for-of 遍历generator
for...of循环可以自动遍历Generator函数,且此时不再需要调用next方法,
一旦next方法的返回对象的done属性为true,for...of循环就会中止,且不包含该返回对象
function *foo() { yield 1; yield 2; yield 3; yield 4; yield 5; return 6;}for (let v of foo()) { console.log(v);}// 1 2 3 4 54. yield* 语句
① 如果yield命令后面跟的是一个遍历器,需要在yield命令后面加上星号,表明它返回的是一个遍历器。这被称为yield*语句。
let a = (function* () { yield 'Hello!'; yield 'Bye!';}());let b = (function* () { yield 'Greetings!'; yield* a; yield 'Ok, bye.';}());for(let value of b) { console.log(value);}结果:
Greetings!Hello!Bye!Ok, bye.② yield命令后面如果不加星号,返回的是整个数组,加了星号就表示返回的是数组的遍历器。
function* gen(){ yield* ["a", "b", "c"];}gen().next() // { value:"a", done:false }遍历嵌套数组:
function* iterTree(tree) { if (Array.isArray(tree)) { for(let i=0; i < tree.length; i++) { yield* iterTree(tree[i]); } } else { yield tree; }}const tree = [ 'a', ['b', 'c'], ['d', 'e'] ];for(let x of iterTree(tree)) { console.log(x);}// a b c d e希望本文所述对大家ECMAScript程序设计有所帮助。
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
ES6新的数组方法、集合、for-of循环、展开运算符(...)甚至异步编程都依赖于迭代器(Iterator)实现。本文会详解ES6的迭代器与生成器,并进一步挖
本文实例讲述了ES6生成器用法。分享给大家供大家参考,具体如下:语法何为生成器?让我们先看看以下代码:function*quips(name){yield"he
前言生成器generator生成器的本质是一个迭代器(iterator)要理解生成器,就要在理解一下迭代,可迭代对象,迭代器,这三个概念Python生成器gen
生成器(generator)概念生成器不会把结果保存在一个系列中,而是保存生成器的状态,在每次进行迭代时返回一个值,直到遇到StopIteration异常结束。
4.生成器(generator)4.1.生成器简介首先请确信,生成器就是一种迭代器。生成器拥有next方法并且行为与迭代器完全相同,这意味着生成器也可以用于Py