【原创】码路工人 Coder-Power

大家好,这里是码路工人有力量,我是码路工人,你们是力量。

github-pages

博客园cnblogs


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 中加入的)

话说语法演化地真是方便啊。在后面的文章中将单独给Promiseasync/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就介绍到这里吧,

这要知道有这个语法糖就可以了,反正以后还真不一定用得到

异步处理主要用到的还是Promiseasync/await

以上。

希望对你能有帮助,下期再见。

-END-  


[ES6系列-07]Generator Function: 生成器函数的更多相关文章

  1. [js高手之路] es6系列教程 - 迭代器与生成器详解

    什么是迭代器? 迭代器是一种特殊对象,这种对象具有以下特点: 1,所有对象都有一个next方法 2,每次调用next方法,都会返回一个对象,该对象包含两个属性,一个是value, 表示下一个将要返回的 ...

  2. [ES6系列-02]Arrow Function:Whats this?(箭头函数及它的this及其它)

    [原创] 码路工人 大家好,这里是码路工人有力量,我是码路工人,你们是力量. 如果没用过CSharp的lambda 表达式,也没有了解过ES6,那第一眼看到这样代码什么感觉? /* eg.0 * fu ...

  3. Nodejs与ES6系列3:generator对象

    3.generator对象 Generator函数是ES6提供的一种异步编程解决方案,语法行为与传统函数完全不同.Generator的中文翻译是生成器,它是ECMAScript6(代号harmory) ...

  4. Generator - Python 生成器

    Generator, python 生成器, 先熟悉一下儿相关定义, generator function 生成器函数, 生成器函数是一个在定义体中存有 'yield' 关键字的函数. 当生成器函数被 ...

  5. [js高手之路] es6系列教程 - 迭代器,生成器,for...of,entries,values,keys等详解

    接着上文[js高手之路] es6系列教程 - 迭代器与生成器详解继续. 在es6中引入了一个新的循环结构for ....of, 主要是用来循环可迭代的对象,那么什么是可迭代的对象呢? 可迭代的对象一般 ...

  6. Python系列之 迭代器和生成器

    很多Python的程序员都会混淆 迭代器 和 生成器 的概念和作用,分不清到底两个有什么区别.今天我们来好好说一说这两个概念. 迭代器(Iterator) Iterator Pattern Itera ...

  7. ES6新特性之生成器函数 (generator function): function*

    一.什么是生成器函数(generator function)? 生成器函数是ES6的新特性之一,它是一个在执行时能中途暂时退出,后面重新调用又能重新进入继续执行的一种函数. 并且在函数内定义的变量的所 ...

  8. ES6笔记(5)-- Generator生成器函数

    系列文章 -- ES6笔记系列 接触过Ajax请求的会遇到过异步调用的问题,为了保证调用顺序的正确性,一般我们会在回调函数中调用,也有用到一些新的解决方案如Promise相关的技术. 在异步编程中,还 ...

  9. ES6生成器函数generator

    ES6生成器函数generator generator是ES6新增的一个特殊函数,通过 function* 声明,函数体内通过 yield 来指明函数的暂停点,该函数返回一个迭代器,并且函数执行到 y ...

随机推荐

  1. Nginx访问日志.Nginx日志切割

    11月27日任务 12.10 Nginx访问日志12.11 Nginx日志切割12.12 静态文件不记录日志和过期时间 1.Nginx访问日志 示例一: 日志格式 vim /usr/local/ngi ...

  2. 全网最简单明了的MySQL连接Eclipse方法(JDBC详细安装方式及简单操作)2020新版

    Step 1 你得有Eclipse 没有出门右拐,我教不了你. Step 2 你得有Mysql MySQL的详细安装过程,我在另一篇博客中给出.戳我 Step 3 安装JDBC 可以去官网下,如果用的 ...

  3. 从零搭建分布式文件系统MinIO比FastDFS要更合适

    前两天跟大家分享了一篇关于如何利用FastDFS组件来自建分布式文件系统的文章,有兴趣的朋友可以阅读下<用asp.net core结合fastdfs打造分布式文件存储系统>.通过留言发现大 ...

  4. 201771030121-王国伟 实验一 软件工程准备—<问题提出与博客首秀>

    项目 内容 课程班级博客链接 https://edu.cnblogs.com/campus/xbsf/nwnu2020SE 本次作业要求链接 https://www.cnblogs.com/nwnu- ...

  5. 使用Hystrix的插件机制,解决在使用线程隔离时,threadlocal的传递问题

    背景 在我们的项目中,比较广泛地使用了ThreadLocal,比如,在filter层,根据token,取到用户信息后,就会放到一个ThreadLocal变量中:在后续的业务处理中,就会直接从当前线程, ...

  6. Istio的流量管理(概念)(istio 系列二)

    Istio的流量管理(概念) 目录 Istio的流量管理(概念) 概述 Virtual services 为什么使用virtual service Virtual services举例 hosts字段 ...

  7. ASP.NET Core Blazor 初探之 Blazor Server

    上周初步对Blazor WebAssembly进行了初步的探索(ASP.NET Core Blazor 初探之 Blazor WebAssembly).这次来看看Blazor Server该怎么玩. ...

  8. transform-translate3d

    translate3d 开启硬件加速,做动效效率比 position 定位置后,改变位置,效果好,比如下拉背景放大效果,上滑的时候背景跟着上滑,可以用 translate3d.亲测效果更好,记下来

  9. go 数组 字符串 切片

    数组 数组定义方式 var a [3]int // 定义长度为3的int型数组, 元素全部为0 var b = [...]int{1, 2, 3} // 定义长度为3的int型数组, 元素为 1, 2 ...

  10. TP5 order排序

    order方法属于模型的连贯操作方法之一,用于对操作的结果排序. ->order('sort desc,id desc') 用法如下: Db::table('think_user')->w ...