Generator函数(一)
Generator函数是ES6提供的一种异步编程解决方案,语法行为与传统函数完全不同。对于这个函数有多种理解。从语法上来理解,可以将它理解成一个状态机,封装了多个内部状态。内部的不同状态是通过yield语句来定义不同的内部状态,return语句也可以看成一个状态。
基本概念
执行Generator函数会返回一个遍历器对象。
//ES6没有规定function关键字与函数名之间的星号写在哪个位置
function* helloWorldGenerator(){
yield 'hello';
yield 'world';
return 'ending';
}
var hw = helloWorldGenerator();
这个函数的调用方式和普通函数一样来调用,调用这个函数,并不执行,返回的也不是函数运行的结果,而是一个指向内部状态的指针对象,也就是一个遍历器对象。
下一步,必须调用next方法,使得指针移向下一个状态。也就是说,每次调用next方法,内部指针就从函数头部或上一次停下来的地方开始执行,直到遇到下一条yield语句(或return语句)为止,
调用next()方法,会返回一个包括value属性和done属性的一个对象。
hw.next();
// {value:'hello',done:false}
//这个value的值就是yield语句的值,done属性的值false表示遍历还没有结束
hw.next();
// {value:'world',done:false}
hw.next();
//{value:'ending',done:true};
hw.next();
//{value:undefined,done:true};
换言之,Generator函数是分段执行的,yield语句是暂停执行的标记,而next方法可以恢复执行。
注意:Generator函数返回的是一个遍历器对象,代表Generator函数的内部指针,以后每次调用这个函数,都会返回带有value和done两个属性的一个对象、
yield语句
1.实际上yield语句是一个暂停标志。
2.yield语句和return语句在Generator函数中的区别
(1)yield语句可以在这个函数中执行多次,我的理解就是,在编写这个函数的时候,可以写多条yield语句,而return语句只能有一条。
(2)yield语句可以使Generator函数获取到多个值,即生成一系列的值。
(3)yield语句可以暂停执行,当执行next方法的时候,可以使函数从上一句yield语句停下的地方开始执行,直到下一个yield语句。而return没有位置记忆功能。
需要注意的是,yield语句后面的表达式,只有当调用next方法、内部指针才会指向该语句时才会执行,因此等于为JavaScript提供手动的“惰性求值”的语法功能。
function* gen(){
yield 123+456;
}
3.Generator函数也可以不用yield语句,这时候变成了一个单纯的暂缓执行函数。
function* f(){
console.log("ffff");
}
var generator = f();
setTimeout(function(){
generator.next();
},2000);
4.还有需要注意的是,yield语句不能用在普通函数中,否则会报错。
var arr = [1,[[2,3],4],[5,6]];
var flat = function* (a){
a.forEach(function(item){
if(typeof item !== 'number'){
yield* flat(item);
}else{
yield item;//这里面不应该使用yield,因为forEach的参数是一个普通函数。
}
})
};
//解决这个错误的方法有
将forEach改为for循环
var arr = [1,[[2,3],4],[5,6]];
var flat = function* (a){
var length = a.length;
for(var i=0;i<length;i++){
var item = a[i];
if(typeof item !== 'number'){
yield* flat(item);
}else{
yield item;
}
}
}
for(var f of flat(arr)){
console.log(f);
//1,2,3,4,5,6
}
5.yield语句用在表达式中,必须要用括号括起来。
6.yield用在函数参数或者是赋值等式的右边时,不需要用括号。
next方法的参数
1.yield语句本身没有返回值,要有值,也是undefined。所以为了赋予yield语句值,所以就让next方法有了参数。
2.next方法有了参数之后,因为Generator函数需要调用next方法来返回yield语句后面的值,这时候调用有参数的next方法之后,这个参数就会被当做上一个yield语句的返回值,注意是yield语句的返回值,而不是yield后面的值。
3.之前说了yield语句可以用于表达式中,所以有参的next方法可以为表达式中的yield语句传值。
function* foo(x){
var y = 2*(yield(x+1));
var z = yield(y/3);
return (x+y+z);
}
var a = foo(5);
a.next();// {value:6,done:false}
a.next(12) // {value:8,done:false}
a.next(13) // {value:42,done:true}
4.注意:由于next方法的参数表示的是上一条yield语句的返回值,所以第一条next语句默认没有参数,V8引擎直接忽略第一次使用next方法时的参数,从语义上来说,第一个next方法用来启动遍历器对象,所以不用带参数。
但是如果实在想在第一条语句中传参,并且能够有输入值,可以在Generator函数外面包一层,相当于先在这个包裹的这一层执行next()方法,用来启动遍历器对象。
//定义一个普通的函数,用来包裹Generator函数。
function wrapper(generatorFunction){
return function(...args){
let generatorObject = generatoraFunction(...args);
generatorObject.next();//先执行这个next
return generatorObject;
}
}
const wrapped = wrapper(function* (){
console.log(`First input:${yield}`);
return 'DONE';
})
wrapped.next('hello');//First input: hello,再执行这个next,并把hello这个值赋给yield
Generator函数(一)的更多相关文章
- ES6 - Note7:Generator函数
Generator函数 1.Generator函数是ES6增加的异步编程解决方案之一,与普通的函数行为完全不同,类似于一个状态机,内部封装了多个状态. 在函数定义的形式上,跟普通函数差不多,有两处不同 ...
- Generator 函数学习笔记
// 使用 function* 定义一个 generator 函数 function* helloWorldGenerator() { yield 'hello'; // yield 关键字作为暂停的 ...
- Generator 函数的含义与用法
Generator 函数是协程在 ES6 的实现,最大特点就是可以交出函数的执行权(即暂停执行). function* gen(x){ var y = yield x + 2; return y; } ...
- generator函数
function* helloWordGenerator() { yield "hello"; yield "world"; return "endi ...
- ES6入门之Generator函数
Generator Generator函数是ES6提供的一种异步编程解决方案,Generator函数是一个状态机,封装了多个内部状态. 执行Generator函数会返回一个遍历器对象,也就是说,Gen ...
- 转: ES6异步编程:Generator 函数的含义与用法
转: ES6异步编程:Generator 函数的含义与用法 异步编程对 JavaScript 语言太重要.JavaScript 只有一根线程,如果没有异步编程,根本没法用,非卡死不可. 以前,异步编程 ...
- Generator函数语法解析
转载请注明出处: Generator函数语法解析 Generator函数是ES6提供的一种异步编程解决方案,语法与传统函数完全不同.以下会介绍一下Generator函数. 写下这篇文章的目的其实很简单 ...
- Generator函数异步应用
转载请注明出处: Generator函数异步应用 上一篇文章详细的介绍了Generator函数的语法,这篇文章来说一下如何使用Generator函数来实现异步编程. 或许用Generator函数来实现 ...
- ES6的generator函数
generator是什么? generator是ES6提供的一种异步编程解决方案,在语法上,可以把它理解为一个状态机,内部封装了多种状态.执行generator,会生成返回一个遍历器对象.返回的遍历器 ...
- ECMAScript 6 入门 ----Generator 函数
本文转自:阮一峰老师的ECMAScript 6 入门,有时间可以看下评论! Generator 函数 简介 基本概念 Generator函数是ES6提供的一种异步编程解决方案,语法行为与传统函数完全不 ...
随机推荐
- python碎片记录(一)
1.python中求幂运算 2**31-1 2的31次方减一(32位整型数据范围为-2^31~2^31-1,最高位为符号位,负向值比正向值绝对值大1) 2.python中整数除法 x=x//1 ...
- TypeError: expected string or buffer的解决方法
错误种类:TypeError: expected string or buffer 具体错误解释:这是因为返回的变量不是字符类型,而导致此错误 具体解决方法:在具体程序段前加if判断语句,判断程序返回 ...
- MFC单文档框架分析及执行流程(转)
原文转自 https://blog.csdn.net/u011619422/article/details/40402705 首先来分析一下MFC单文档类的结构: 它包括如下几个类: CAboutDl ...
- write-ups
https://github.com/MarioVilas/write-ups https://github.com/Deplorable-Mountaineer/Robot_Dynamite htt ...
- problems when installed mysql in linux ubuntu
reference:http://www.jb51.net/article/87160.htm?pc 1.ERROR 2002 (HY000): Can't connect to local MySQ ...
- C核心 那些个关键字
概述 - C语言老了 目前而言(2017年5月12日) C语言中有 32 + 5 + 7 = 44 个关键字. 具体如下 O(∩_∩)O哈哈~ -> C89关键字 char short int ...
- [How to]Cloudera manager 离线安装手册
2016-01-1910:54:05 增加kafka 1.简介 本文介绍在离线环境下安装Cloudera manager和简单使用方法 2.环境 OS:CentOS 6.7 Cloudera man ...
- Leetcode 之Balanced Binary Tree(49)
用递归的方式来做,左右两棵子树的高度差不超过1.分成两部分,一部分递归得到树的高度,一部分递归检查左右子树是否是平衡二叉树. int getHeight(TreeNode *root) { ; ; } ...
- HDU-5280
Senior's Array Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others)To ...
- 在k8s中安装flannel的故障解决: Failed to create SubnetManager: error retrieving pod spec for : the server does not allow access to the requested resource
花了一个上午来追踪问题,k8s都反复新建了十多次,docker都重启了几次.(一次显示不有获取磁盘空间,重启docker,清空存储解决) 在用kubeadm安装容器化的几个组件时,flannel组件死 ...