[ES6系列-07]Generator Function: 生成器函数
【原创】码路工人 Coder-Power
大家好,这里是码路工人有力量,我是码路工人,你们是力量。
Generator function 生成器函数
是ES6
中新增的语法糖,本质上讲,就是以封装成一个遍历器的形式,让编码的你获得程序的执行控制权,通俗地说就是,流程控制上,踹一脚,走一步,不要太暴力~
0.前言
要说到生成器函数,就不得不提到javascript
的异步编程方式演进史。
(不能跑题不能跑题不能跑题)
- 1.普通的回调函数方式(callback)
- 2.事件/发布--订阅者模式(event/publisher--observer)
- 3.Promise
- 4.Generator
- 5.async/await(在ES8/ES2017中正式提出)
其中Promise
是一个里程碑,解决了回调函数嵌套时的callback hell
回调地狱问题(层级嵌套太多难以阅读与维护)。
而本文主角Generator
更像是作为一个过渡语法,在推出async/await
后就基本很少用了。
async/await
,CSharper转前端,一看就像见到亲人,C#
中有同样的语法。(#注:C# 5.0 中加入的)
话说语法演化地真是方便啊。在后面的文章中将单独给Promise
和async/await
开贴。
1.一句话介绍你自己
1.1 Talk is cheep, show you the CODE!
/* eg.0
* Simple Example of Generator-Function
*/
//----------------------------------------
let songs = ["Hero", "Here I am", "The Show"]
// Generator-Function is Here Below:
function *play(songs) {
for(let i=0; i<songs.length; i++) {
yield songs[i]
}
// 这里可以有return
}
let g = play(songs)
let next = g.next()
console.log(next) // { value: 'Hero', done: false }
next = g.next()
console.log(next) // { value: 'Here I am', done: false }
next = g.next()
console.log(next) // { value: 'The Show', done: false }
next = g.next()
console.log(next) // { value: undefined, done: true }
//----------------------------------------
1.2 关于生成器函数的定义
第一,函数定义处有个
*
符号。*
既可以紧跟在function
后面,也可以贴在函数名前。第二,函数内有
yield
关键字用以中断处理流程,并可以临时对外提供一个返回值。
其它,写法上与普通函数无异
1.3 关于生成器函数的使用
1.像普通函数一样调用,得到的不是任何具体的返回值,而是一个迭代器。(也可以看作是一个状态机)
2.来一脚试试。通过调用得到的迭代器对象上的
next()
方法,开始得到第一个返回值对象。3.一直踹。每一次调用
next()
方法,得到一个结果对象,可以看到上面代码注释中的打印信息。其中
value
即当前获得的值,done
即迭代状态,它只有ture/false
两种可能,当它变为true
,即迭代完成。
2.主流使用方式
正常的使用方式也即良好的实践吧,应该是配合promise
来使用的。
来一个简陋的示例:
/* eg.1
* Simple Example of Generator-Function
*/
//----------------------------------------
function* func() {
// 这里应该是ajax请求,结果是一个promise对象,简单模拟一下
yield new Promise(function(resolve,reject){
resolve('Hello')
})
// 这里应该是ajax请求,结果是一个promise对象,简单模拟一下
yield new Promise(function(resolve,reject){
resolve('World')
})
}
let g = func()
g.next().value.then((data) => {
console.log('log-1:', data)
return g.next().value
}).then((data) => {
console.log('log-2:', data)
})
// log-1: Hello
// log-2: World
//----------------------------------------
这样,就将异步的ajax
处理简单地以同步的样子书写出来了。
3.粗陋的实践
除了可以处理异步,码路工人个人觉得,Generator
比较适合用在循环处理的场景。其中就实践过类似下面这个处理的例子:
/* eg.2 (part1)
* My Example of Generator-Function
*/
//----------------------------------------
// 以下代码中都省略掉了检测处理
// 首先来一个Gernerator函数
function *getDateInPeriod(from, to, includEnd=false) {
let fromDate = new Date(from)
let toDate = new Date(to)
if(includEnd) {
toDate = addDays(toDate, 1)
}
for(let date = fromDate;date < toDate; date=addDays(date, 1)) {
yield date
}
}
// 上面调用到了工具函数addDays
function addDays(date, days){
date = new Date(date.setDate(date.getDate() + days))
return date
}
//----------------------------------------
然后就是使用了,在循环取值的过程中,会有一些业务处理(同步的或异步的)。
/* eg.2 (part2)
* My Example of Generator-Function
*/
//----------------------------------------
let arrDate = getDateInPeriod('2019-06-06', '2019-06-30', true)
let daysInPeriod = arrDate.next()
while(!daysInPeriod.done) {
// 这里是一堆其它业务处理
console.log(daysInperiod)
daysInPeriod = arrDate.next()
}
console.log(daysInperiod)
//----------------------------------------
打印信息结果:
{ value: 2019-06-06T00:00:00.000Z, done: false }
{ value: 2019-06-07T00:00:00.000Z, done: false }
...
{ value: 2019-06-29T00:00:00.000Z, done: false }
{ value: 2019-06-30T00:00:00.000Z, done: false }
{ value: undefined, done: true }
上面的例子中,还是同步的方式。也许不是好的用法,但它能告诉你这是一种用法。
4.听说过的实践
这个码工没有实践过,大意是,Generator
函数这个语法糖是按照yield
将代码分成多个部分,人手工控制执行每一部分,于是,封装一个执行器函数,简化流程控制,使异步编程轻松愉快。
其实,在有了async/await
后真的没有这个必要了。
(co.js
就是一个这种Generator
的执行库)
4.其它
除了上面介绍的生成器函数的主要用法,其实还有点其它小特性,码路工人并没有实践过,感觉也不怎么会用到,这里稍作了解即可。
4.1 可以接收由next
传递进的参数。
/* eg.3
* parameters
*/
//----------------------------------------
function *genFunc(p) {
console.log(`p is ${p}`)
let p1 = yield p
console.log(`p1 is ${p1}`)
let p2 = yield p1
console.log(`p2 is ${p2}`)
}
let gen = genFunc(1)
// 第一次next无法传参到Generator函数,应该由最初的函数调用处传递
gen.next(2)
// 从第二步next() 传的参数,在第一个yield处接收
gen.next(3)
关于给yield
传参确实看起来稍微有一点诡异,其实理解了也就不觉得奇怪了。
说明已贴在上面的注释中。
4.2 可以有默认的迭代器了
/* eg.4
* get value by for-of loop
*/
//----------------------------------------
function *foo() {
for(let i=0; i<6; i++) {
yield i
}
}
let gen = foo()
for(let item of gen) {
console.log(item)
}
// 打印结果:
// 0
// 1
// 2
// 3
// 4
// 5
//----------------------------------------
总结
关于生成器函数Generator Function
就介绍到这里吧,
这要知道有这个语法糖就可以了,反正以后还真不一定用得到
异步处理主要用到的还是Promise
跟async/await
。
以上。
希望对你能有帮助,下期再见。
-END-
[ES6系列-07]Generator Function: 生成器函数的更多相关文章
- [js高手之路] es6系列教程 - 迭代器与生成器详解
什么是迭代器? 迭代器是一种特殊对象,这种对象具有以下特点: 1,所有对象都有一个next方法 2,每次调用next方法,都会返回一个对象,该对象包含两个属性,一个是value, 表示下一个将要返回的 ...
- [ES6系列-02]Arrow Function:Whats this?(箭头函数及它的this及其它)
[原创] 码路工人 大家好,这里是码路工人有力量,我是码路工人,你们是力量. 如果没用过CSharp的lambda 表达式,也没有了解过ES6,那第一眼看到这样代码什么感觉? /* eg.0 * fu ...
- Nodejs与ES6系列3:generator对象
3.generator对象 Generator函数是ES6提供的一种异步编程解决方案,语法行为与传统函数完全不同.Generator的中文翻译是生成器,它是ECMAScript6(代号harmory) ...
- Generator - Python 生成器
Generator, python 生成器, 先熟悉一下儿相关定义, generator function 生成器函数, 生成器函数是一个在定义体中存有 'yield' 关键字的函数. 当生成器函数被 ...
- [js高手之路] es6系列教程 - 迭代器,生成器,for...of,entries,values,keys等详解
接着上文[js高手之路] es6系列教程 - 迭代器与生成器详解继续. 在es6中引入了一个新的循环结构for ....of, 主要是用来循环可迭代的对象,那么什么是可迭代的对象呢? 可迭代的对象一般 ...
- Python系列之 迭代器和生成器
很多Python的程序员都会混淆 迭代器 和 生成器 的概念和作用,分不清到底两个有什么区别.今天我们来好好说一说这两个概念. 迭代器(Iterator) Iterator Pattern Itera ...
- ES6新特性之生成器函数 (generator function): function*
一.什么是生成器函数(generator function)? 生成器函数是ES6的新特性之一,它是一个在执行时能中途暂时退出,后面重新调用又能重新进入继续执行的一种函数. 并且在函数内定义的变量的所 ...
- ES6笔记(5)-- Generator生成器函数
系列文章 -- ES6笔记系列 接触过Ajax请求的会遇到过异步调用的问题,为了保证调用顺序的正确性,一般我们会在回调函数中调用,也有用到一些新的解决方案如Promise相关的技术. 在异步编程中,还 ...
- ES6生成器函数generator
ES6生成器函数generator generator是ES6新增的一个特殊函数,通过 function* 声明,函数体内通过 yield 来指明函数的暂停点,该函数返回一个迭代器,并且函数执行到 y ...
随机推荐
- CentOS7.x上轻量级TCP转发工具rinetd的安装配置
一.实验背景 Linux下端口转发一般都使用iptables来实现,使用iptables可以很容易将TCP和UDP端口从防火墙转发到内部主机上. 如果需要将流量从专用地址转发到不在您当前网络上的机器上 ...
- DP背包(一)
01背包 for(int i=0;i<n;i++) //遍历每一件物品 for(int j=v;j>=wei[i];j--)//遍历背包容量,表示在上一层的基础上,容量为J时,第i件物品装 ...
- vue获取当前时间 实时刷新
需求:获取当前系统时间,在页面上展示 年月日 时分秒 ,并且实时刷新,和系统时间保持一致 第一步:在deta 里面声明两个变量第二步:把时间调用写在created() 生命周期里面,进入页面就需要调用 ...
- Pandas切片操作:很容易忽视的SettingWithCopyWarning
Pandas是一个强大的分析结构化数据的工具集,主要用于数据挖掘和数据分析,同时也提供数据清洗功能. 很多初学者在数据的选取,修改和切片时经常面临一些困惑.这是因为Pandas提供了太多方法可以做同样 ...
- Java——类的访问修饰符
1.java中外部类的访问修饰符有如下四种: public,默认,abstract,final // public,默认,abstract,final. public class Test1 {} c ...
- 模块(类)之间解耦利器:EventPublishSubscribeUtils 事件发布订阅工具类
如果熟悉C#语言的小伙伴们一般都会知道委托.事件的好处,只需在某个类中提前定义好公开的委托或事件(委托的特殊表现形式)变量,然后在其它类中就可以很随意的订阅该委托或事件,当委托或事件被触发执行时,会自 ...
- Edge Weight Assignment(树-异或-贪心)
大意: 给定一棵无根树,要求你任意设置n-1条边的边权. 使得任意叶子节点间边权的XOR值为0: 此时,令f为所有边权数值不同的个数,求最小的f和最大的f. \(\color{Red}{------- ...
- P2422 良好的感觉(两头单调)
描述:https://www.luogu.com.cn/problem/P2422 kkk做了一个人体感觉分析器.每一天,人都有一个感受值Ai,Ai越大,表示人感觉越舒适.在一段时间[i, j]内,人 ...
- HC32F003C4PA GPIO Output
1.打开启动文件,找到并跳转至SystemInit函数 void SystemInit(void) { stc_clk_systickcfg_t stcCfg; // TODO load trim f ...
- Qt数据库总结
使用Qt SQL库 头文件: #include <QtSql> 项目: QT += sql 常用类 QSqlDatabase:数据库的连接打开等操作 QSqlQuery:执行语句,获取结果 ...