读深入ES6记[二]

ES6

第三章:生成器(Generator)

1.解决的问题

当前可以总结出来的是 简化代码解决回调地狱 ,举个栗子呗,OK的

  • 简化代码例子:

还记得《读深入ES6[一]》中为普通对象增加迭代器遍历数据的代码吗?

Object.prototype[Symbol.iterator] = function() {
  return this; // 返回迭代器对象。一个迭代器必须实现了next接口,返回值格式为{done: [bool], value: }
};
Object.prototype.next = function () {
  this.currentIdx = (this.currentIdx === undefined ? -1 : this.currentIdx) + 1;
  let keys = Object.keys(this);
  let keyTotal = keys.length - 1; // 排除掉currentIdx这个属性
  let done = keyTotal < this.currentIdx + 1;
  let key = done ? null : keys[this.currentIdx];
  let value = key ? this[key] : undefined;
  return {done: done, value: value};
};

如果使用生成器来实现,会是以下这样:

Object.prototype[Symbol.iterator] = function* () {
  let keys = Object.keys(this);
  for (let key of keys) {
    yield this[key];
  }
};

因为生成器函数返回的就是一个迭代器对象,而且自实现了next()的接口方法。

  • 解决回调地狱例子:

从最原始的回调函数地狱般嵌套,到后来Promise的解决方案,都没能很好地以人类同步的思维来书写异步代码。
自从有了Generator,配合上Promise和一些第三方库的方法,如Q.async,可以使异步编程代码更加接近人脑思维。
如下代码,会顺序执行一个一个异步方法(用Promise实现),有没感叹异步编程从未有过如此优雅的代码哈

Q.async(function* () {

  yield first_async();

  yield second_async();

  yield third_async();

});

题外话,有没想了解Q.async做了些啥呢?其实很简单,就是在异步回调的时候执行生成器的.next()方法罢了,源码伺候:

function async(makeGenerator) {
  return function () {
    // when verb is "send", arg is a value
    // when verb is "throw", arg is an exception
    function continuer(verb, arg) {
      var result;

      // Until V8 3.19 / Chromium 29 is released, SpiderMonkey is the only
      // engine that has a deployed base of browsers that support generators.
      // However, SM's generators use the Python-inspired semantics of
      // outdated ES6 drafts.  We would like to support ES6, but we'd also
      // like to make it possible to use generators in deployed browsers, so
      // we also support Python-style generators.  At some point we can remove
      // this block.

      if (typeof StopIteration === "undefined") {
        // ES6 Generators
        try {
          result = generator[verb](arg);
        } catch (exception) {
          return reject(exception);
        }
        if (result.done) {
          return Q(result.value);
        } else {
          return when(result.value, callback, errback);
        }
      } else {
        // SpiderMonkey Generators
        // FIXME: Remove this case when SM does ES6 generators.
        try {
          result = generator[verb](arg);
        } catch (exception) {
          if (isStopIteration(exception)) {
            return Q(exception.value);
          } else {
            return reject(exception);
          }
        }
        return when(result, callback, errback);
      }
    }
    var generator = makeGenerator.apply(this, arguments);
    var callback = continuer.bind(continuer, "next");
    var errback = continuer.bind(continuer, "throw");
    return callback();
  };
}

2.与普通函数的区别

  1. 声明方式:
    生成器函数用function*;普通函数用function

  2. 返回方式:
    生成器函数用yield关键字,可多次调用,每次调用都自暂停(这个强大);普通函数用return关键字,只可调用一次,然后就结束

  3. 返回值:
    生成器函数返回一个迭代器对象,这里将它称为生成器对象;普通函数返回return的值(如没return则返回undifined)

3.执行原理

它并非另外开线程,它仍然运行在Javascript的主线程之上。
每当生成器执行yield语句时,生成器的堆栈结构(参数,本地变量,临时值,生成器内部当前的执行位置)被移出堆栈。然而生成器对象保留了对这个堆栈结构的引用,所以稍后调用.next()可以重新激活堆栈结构并且继续执行


第四章:模板字符串

1.反撇号``

对于简单的字符串拼接,再也不用第三方工具类的模板方法了(如lodash,underscore的_.template())

var data = {name: 'daniel'};
console.log(`Hello ${data.name}`);

2.特性

  • 可写多行
var data = {name: 'daniel'};
console.log(`
  Dear ${data.name},
       Some message here
`);
  • 任意JS表达式
var data = {name: 'daniel'};
function getBody() {
  return 'Some message here'
}
var hiTmpl = `Dear ${data.name}`;
console.log(`
  ${hiTmpl},
       ${getBody()}
`);

3.局限性

无特殊字符转义,无国际化,无内建循环语法,无条件语句支持等
所以暂把它用于普通字符串模板的处理,如错误消息。页面模板还是交由专业的模板引擎处理

4.标签模板(tagged template)突破局限

标签模板可以自定义模板处理方法,通过在字符串模板前加一个标签指定处理函数,如下:

var name = 'Daniel';
var str = CustomTag`Hello ${name}, nice to see you.`;

然后我们就可以声明处理函数CustomTag
templateData是一个根据占位符分隔开的字符串数组,arguments从索引1开始都是JS表达式的值,所以此时你拥有了绝对的控制权,发挥你的小宇宙吧

function CustomTag(templateData) {
  var result = '';
  // templateData [ 'Hello ', ', nice to see you.' ]
  // arguments[1]: 'Daniel'
  // 加一些代码尽情处理吧
  return result;
}

这就是第三,四章的学习情况,接下来可继续看《读深入ES6记[三]》

前面章节的学习情况请看:
《读深入ES6记[一]》

--EOF--

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,636评论 5 468
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,890评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,680评论 0 330
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,766评论 1 271
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,665评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,045评论 1 276
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,515评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,182评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,334评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,274评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,319评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,002评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,599评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,675评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,917评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,309评论 2 345
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,885评论 2 341

推荐阅读更多精彩内容