轻松学会ES6新特性之生成器
生成器虽然是ES6最具魔性的新特性,但也是最难懂得的一节,笔者写了大量的实例来具体化这种抽象的概念,能够让人一看就懂,目的是希望别人不要重复或者减少笔者学习生成器的痛苦经历。
在说具体的ES6生成器之前,我觉得我们应该先来看一个例子,因为它真的很晦涩,我们需要先增加一下兴趣会说话的猫
ES6生成器可以说是ES6新特性中最具魔力的特性,它可以极大的简化代码,并且贯穿了整个ES6的使用。它与我们以前学过的js特性都不同,它可以在函数的内部随时暂停。
那么我们从最简单的定义生成器说起,这可能会相对容易一些,以下是会说话的猫的生成器代码:
- function* quips(name) {
- yield "hello " + name + "!";
- yield "i hope you are enjoying the blog posts";
- if (name.toLowerCase().startsWith("x")) {
- yield "hey, it's cool how your name starts with an X, " + name;
- }
- yield "see you later!";
- }
它看起来像一个函数,但是也有一些不同
- 它使用function*来声明,而不是function。
- 它使用yield来返回,而不是return。生成器函数可以yield多次而不会退出程序,在执行过程中遇到yield表达式立刻暂停,后续可恢复执行状态,而return直接退出程序。
- var meow_iter = quips("Xingu");
- //[object Generator]
- meow_iter.next();
- //{value:"hello Xingu!",done:false}
- meow_iter.next();
- //{value:"i hope you are enjoying the blog posts",done:false}
- meow_iter.next();
- //{value:"hey, it's cool how your name starts with an X, Xingu",done:false}
- meow_iter.next();
- //{value:"see you later!",done:false}
- meow_iter.next();
- //{value:undefied,done:true}
以上就是会说话的猫生成器的执行过程,看过ES6迭代器的同学可能基本上能看懂这段代码了。但是当你调用一个生成器的时候,它并非立即执行,而是在生成器函数第一句代码执行之前返回一个生成器对象(就是上面的meow_iter),这个时候函数冻结了。当你调用对象的.next()方法的时候,函数调用将其自身解冻并一直运行到下一个yield表达式,再次暂停。 当调用最后一个.next()的时候,返回的done为true,调用结束。这就是基本的生成器执行过程了。
- function* num(){
- yield 1;
- yield 2;
- return 3;
- yield 4;
- }
- var itr = num();
- console.log(itr.next());
- console.log(itr.next());
- console.log(itr.next());
- console.log(itr.next());
- console.log(itr.next());
- // Object { value: 1, done: false }
- // Object { value: 2, done: false }
- // Object { value: 3, done: true }
- // Object { value: undefined, done: true }
- // Object { value: undefined, done: true }
事实胜于雄辩,当我们在生成器中使用return的时候,那么return的值将会作为最后的返回值,而return后面的yield语句不会生效。
- function* num(){
- yield 1;
- yield 2;
- return 3;
- yield 4;
- }
- for(var i of num()){
- console.log(i);
- }
- //1
- //2
可以看到for-of并不会遍历到return的值,也不会遍历到return之后yield的值。
- function* num(){
- console.log("start");
- for(var i = 0; i < 3; i++){
- yield i;
- }
- console.log("end");
- }
- for(var i of num()){
- console.log(i);
- }
- // start
- // 0
- // 1
- // 2
- // end
官方对与循环中yield是这样解释的:在循环中,每当生成器执行yields语句,生成器的堆栈结构(本地变量、参数、临时值、生成器内部当前的执行位置)被移出堆栈。然而,生成器对象保留了对这个堆栈结构的引用(备份),所以稍后调用.next()可以重新激活堆栈结构并且继续执行。什么东西?好像很高深的样子。说了这么长一串,无非就是说循环中的yield和正常语句的yield在使用上没有区别。
- function* num(){
- yield 1;
- yield 2;
- }
- function* num1(){
- yield 3;
- yield* num();
- yield 4
- }
- for(var i of num1()){
- console.log(i);
- }
- // 3
- // 1
- // 2
- // 4
.next()传参
- function* add(num){
- var x = yield (num + 1);
- console.log(x);
- var y = yield (x + 1);
- return x + y;
- }
- var a = add(1);
- console.log(a.next());
- console.log(a.next());
- console.log(a.next());
- // Object { value: 2, done: false }
- // undefined
- // Object { value: NaN, done: false }
- // Object { value: NaN, done: true }
这是一个传入参数的例子
- function* add(num){
- var x = yield (num + 1);
- console.log(x);
- var y = yield (x + 1);
- return x + y;
- }
- var a = add(1);
- console.log(a.next());
- console.log(a.next(2));
- console.log(a.next(3));
- // Object { value: 2, done: false }
- // 2
- // Object { value: 3, done: false }
- // Object { value: 5, done: true }
可见,我们在生成器中yield的返回值不能初始化左边的变量,实际上给.next()方法传参数, 那么这个参数将会作为上一次yield语句的返回值,如上我们传入2作为参数,那么上一个yield(num+1)就相当于返回了2,不然我们无法使用它的返回值,如例1。这样说还是有点难懂,一些复杂的问题往往因为一些简单的举例而变得容易理解。实际上这种给.next()传参的做法常常用来代替异步方式的Ajax请求,看完下面的例子你可能会恍然大悟。
- "use strict";
- function* main() {
- var result = yield request("http://www.filltext.com?rows=10&f={firstName}");
- console.log(result);
- //do 别的ajax请求;
- }
- function request(url) {
- var r = new XMLHttpRequest();
- r.open("GET", url, true);
- r.onreadystatechange = function () {
- if (r.readyState != 4 || r.status != 200) return;
- var data = JSON.parse(r.responseText);
- //数据成功返回以后, 代码就能够继续往下走了;
- it.next(data);
- };
- r.send();
- }
- var it = main();
- it.next();
- console.log("执行到这儿啦");
it.next(data)将data作为上一次yield的返回值赋值给result,然后打印result,这种用同步的方式实现了和异步编程一样的效果,真是令人惊讶!
- function* producevalues() {
- setup();
- try {
- // ... 生成一些值
- } finally {
- cleanup();
- }
- }
- for (var value of producevalues()) {
- work(value);
- }
我们希望无论如何都在cleanup()中释放一些资源,但这里有一些情况:
- 我们没在 try 代码块中调用 work(value) ,如果它抛出异常怎么办?
- 或者假设 for-of 循环包含一条 break 语句或 return 语句。清理步骤又会如何执行呢?
- function* num() {
- try {
- for(var i = 0; i < 3; i++){
- yield i;
- }
- } finally {
- console.log("finally run");
- }
- }
- function work(value){
- if(value != 1){
- console.log(value);
- }else{
- throw(new Error("error"));
- }
- }
- for(var value of num()) {
- work(value);
- }
- // 0
- // finally run
- // Error: error
循环中抛出异常,finally照常执行
- function* num() {
- try {
- for(var i = 0; i < 3; i++){
- yield i;
- }
- } finally {
- console.log("finally run");
- }
- }
- function work(value){
- console.log(value);
- }
- for(var value of num()) {
- if(value == 1){
- break;
- }
- work(value);
- }
- // 0
- // finally run
循环中break或return,finally照常执行。
- function* num() {
- try {
- for(var i = 0; i < 3; i++){
- yield i;
- }
- } finally {
- console.log("finally run");
- }
- }
- function work(value){
- console.log(value);
- }
- var a = num();
- for(var value of a) {
- if(value == 1){
- a.return();
- }
- work(value);
- }
- // 0
- // finally run
- // 1
下面我们介绍.throw()方法
- function* num(){
- try{
- yield 1
- }catch(e){
- console.log("内部捕获")
- }
- }
- var a = num();
- a.next();
- try{
- a.throw("a");
- }catch(e){
- console.log("外部捕获")
- }
- //内部捕获
- function* num(){
- try{
- yield 1
- }catch(e){
- console.log("内部捕获")
- }
- yield 2
- }
- var a = num();
- a.next();
- a.next();
- try{
- a.throw("a");
- }catch(e){
- console.log("外部捕获")
- }
- //外部捕获
最后补充一点关于生成器中的this以及它的原型
- 生成器中的this就是它的调用者。
- 生成器生成的Iterator,不但继承了Iterator的原型, 也继承了Generator的原型。
轻松学会ES6新特性之生成器的更多相关文章
- ES6新特性之生成器函数 (generator function): function*
一.什么是生成器函数(generator function)? 生成器函数是ES6的新特性之一,它是一个在执行时能中途暂时退出,后面重新调用又能重新进入继续执行的一种函数. 并且在函数内定义的变量的所 ...
- ES6新特性三: Generator(生成器)函数详解
本文实例讲述了ES6新特性三: Generator(生成器)函数.分享给大家供大家参考,具体如下: 1. 简介 ① 理解:可以把它理解成一个函数的内部状态的遍历器,每调用一次,函数的内部状态发生一次改 ...
- ES6新特性概览
本文基于lukehoban/es6features ,同时参考了大量博客资料,具体见文末引用. ES6(ECMAScript 6)是即将到来的新版本JavaScript语言的标准,代号harmony( ...
- Atitit js版本es5 es6新特性
Atitit js版本es5 es6新特性 Es5( es5 其实就是adobe action script的标准化)1 es6新特性1 Es5( es5 其实就是adobe action scrip ...
- ES6新特性简介
ES6新特性简介 环境安装 npm install -g babel npm install -g babel-node //提供基于node的REPL环境 //创建 .babelrc 文件 {&qu ...
- 必须掌握的ES6新特性
ES6(ECMAScript2015)的出现,让前端开发者收到一份惊喜,它简洁的新语法.强大的新特性,带给我们更便捷和顺畅的编码体验,赞! 以下是ES6排名前十的最佳特性列表(排名不分先后): 1.D ...
- 你不知道的JavaScript--Item24 ES6新特性概览
ES6新特性概览 本文基于lukehoban/es6features ,同时参考了大量博客资料,具体见文末引用. ES6(ECMAScript 6)是即将到来的新版本JavaScript语言的标准,代 ...
- 34.js----JS 开发者必须知道的十个 ES6 新特性
JS 开发者必须知道的十个 ES6 新特性 这是为忙碌的开发者准备的ES6中最棒的十个特性(无特定顺序): 默认参数 模版表达式 多行字符串 拆包表达式 改进的对象表达式 箭头函数 =&> ...
- ES6新特性概览1
本文基于lukehoban/es6features ,同时参考了大量博客资料,具体见文末引用. ES6(ECMAScript 6)是即将到来的新版本JavaScript语言的标准,代号harmony( ...
随机推荐
- 14.LINUX-platform机制实现驱动层分离(详解)
版权声明:本文为博主原创文章,未经博主允许不得转载. 本节目标: 学习platform机制,如何实现驱动层分离 1.先来看看我们之前分析输入子系统的分层概念,如下图所示: 如上图所示,分 ...
- 异常 SQLErrorCodes loaded: [DB2, Derby, H2, HSQL, Informix, MS-SQL, MySQL, Oracle, PostgreSQL, Sybase]
使用Spring 的JDBCtemplate 调用数据库的时候 出现了如下的问题 SQLErrorCodes loaded: [DB2, Derby, H2, HSQL, Informix, MS-S ...
- Servlet学习应该注意的几点
一.Servlet生命周期(即运行过程) (1)初始阶段,调用init()方法 (2)响应客户请求阶段,调用service()方法.由service()方法根据提交方式不同执行doGet()或doPo ...
- Struts2的核心运行流程,原理图解
感觉很有必要制定一个计划,这样盲目的想到哪里写到哪里,不符合我大程序员的思维逻辑呀~~~嗯...那就从基本的开始吧,循循渐进,今天想到的先写上,不能浪费了,哈哈哈................... ...
- macbook 263企业邮箱设置
第一步:打开邮箱,点击添加账号,选择其他 第二步:填写完整的电子邮件地址和密码 第三步:填写收件服务器(popcom.263xmail.com),发件服务器(smtpcom.263xmail.com)
- ThinkPHP中foreach和volist的区别
1.foreach标签foreach标签用于循环输出:foreach(name,item,key)name(必须):要输出的数据模板变量item(必须):循环单原变量key(可选):循环的key变量, ...
- swiper拖拽之后不自动滑动问题
//swiper轮播图 var mySwiper = new Swiper('.swiper-container',{ initialSlide :0, autoplay : 3000, direct ...
- fitnesse - 一个简单的例子(slim)
fitnesse - 一个简单的例子(slim) 2017-09-30 目录1 编写测试代码(Fixture code)2 编写wiki page并运行 2.1 新建wikiPage 2.2 运行 ...
- 浅谈Java抽象类
什么是抽象类?这名字听着就挺抽象的,第一次听到这个名字还真有可能被唬住.但是,就像老人家所说的,一切反动派都是纸老虎,一切有着装x名字的概念也是纸老虎.好吧,我们已经从战略上做到了藐视它,现在就要战术 ...
- Elasticsearch-sql 用SQL查询Elasticsearch
Elasticsearch的查询语言(DSL)真是不好写,偏偏查询的功能千奇百怪,filter/query/match/agg/geo各种各样,不管你是通过封装JSON还是通过python/java的 ...