成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

ES6 系列之 Babel 將 Generator 編譯成了什么樣子

EddieChan / 2355人閱讀

摘要:前言本文就是簡單介紹下語法編譯后的代碼。如果有錯誤或者不嚴謹?shù)牡胤?,請?wù)必給予指正,十分感謝。如果喜歡或者有所啟發(fā),歡迎,對作者也是一種鼓勵。

前言

本文就是簡單介紹下 Generator 語法編譯后的代碼。

Generator
function* helloWorldGenerator() {
  yield "hello";
  yield "world";
  return "ending";
}

我們打印下執(zhí)行的結(jié)果:

var hw = helloWorldGenerator();

console.log(hw.next()); // {value: "hello", done: false}
console.log(hw.next()); // {value: "world", done: false}
console.log(hw.next()); // {value: "ending", done: true}
console.log(hw.next()); // {value: undefined, done: true}
Babel

具體的執(zhí)行過程就不說了,我們直接在 Babel 官網(wǎng)的 Try it out 粘貼上述代碼,然后查看代碼被編譯成了什么樣子:

/**
 * 我們就稱呼這個版本為簡單編譯版本吧
 */
var _marked = /*#__PURE__*/ regeneratorRuntime.mark(helloWorldGenerator);

function helloWorldGenerator() {
  return regeneratorRuntime.wrap(
    function helloWorldGenerator$(_context) {
      while (1) {
        switch ((_context.prev = _context.next)) {
          case 0:
            _context.next = 2;
            return "hello";

          case 2:
            _context.next = 4;
            return "world";

          case 4:
            return _context.abrupt("return", "ending");

          case 5:
          case "end":
            return _context.stop();
        }
      }
    },
    _marked,
    this
  );
}

猛一看,好像編譯后的代碼還蠻少的,但是細細一看,編譯后的代碼肯定是不能用的呀,regeneratorRuntime 是個什么鬼?哪里有聲明呀?markwrap 方法又都做了什么?

難道就不能編譯一個完整可用的代碼嗎?

regenerator

如果你想看到完整可用的代碼,你可以使用 regenerator,這是 facebook 下的一個工具,用于編譯 ES6 的 generator 函數(shù)。

我們先安裝一下 regenerator:

npm install -g regenerator

然后新建一個 generator.js 文件,里面的代碼就是文章最一開始的代碼,我們執(zhí)行命令:

regenerator --include-runtime generator.js > generator-es5.js

我們就可以在 generator-es5.js 文件看到編譯后的完整可用的代碼。

而這一編譯就編譯了 700 多行…… 編譯后的代碼可以查看 generator-es5.js

總之編譯后的代碼還蠻復雜,我們可以從中抽離出大致的邏輯,至少讓簡單編譯的那段代碼能夠跑起來。

mark 函數(shù)

簡單編譯后的代碼第一段是這樣的:

var _marked = /*#__PURE__*/ regeneratorRuntime.mark(helloWorldGenerator);

我們查看完整編譯版本中 mark 函數(shù)的源碼:

runtime.mark = function(genFun) {
  genFun.__proto__ = GeneratorFunctionPrototype;
  genFun.prototype = Object.create(Gp);
  return genFun;
};

這其中又涉及了 GeneratorFunctionPrototype 和 Gp 變量,我們也查看下對應(yīng)的代碼:

function Generator() {}
function GeneratorFunction() {}
function GeneratorFunctionPrototype() {}

...

var Gp = GeneratorFunctionPrototype.prototype =
  Generator.prototype = Object.create(IteratorPrototype);

GeneratorFunction.prototype = Gp.constructor = GeneratorFunctionPrototype;

GeneratorFunctionPrototype.constructor = GeneratorFunction;

GeneratorFunctionPrototype[toStringTagSymbol] =
  GeneratorFunction.displayName = "GeneratorFunction";

這段代碼構(gòu)建了一堆看起來很復雜的關(guān)系鏈,其實這是參照著 ES6 規(guī)范構(gòu)建的關(guān)系鏈:

圖中 +@@toStringTag:s = "Generator" 的就是 Gp,+@@toStringTag:s = "GeneratorFunction" 的就是 GeneratorFunctionPrototype。

構(gòu)建關(guān)系鏈的目的在于判斷關(guān)系的時候能夠跟原生的保持一致,就比如:

function* f() {}
var g = f();
console.log(g.__proto__ === f.prototype); // true
console.log(g.__proto__.__proto__ === f.__proto__.prototype); // true

為了簡化起見,我們可以把 Gp 先設(shè)置為一個空對象,不過正如你在上圖中看到的,next()、 throw()、return() 函數(shù)都是掛載在 Gp 對象上,實際上,在完整的編譯代碼中,確實有為 Gp 添加這三個函數(shù)的方法:

// 117 行
function defineIteratorMethods(prototype) {
  ["next", "throw", "return"].forEach(function(method) {
    prototype[method] = function(arg) {
      return this._invoke(method, arg);
    };
  });
}

// 406 行
defineIteratorMethods(Gp);

為了簡單起見,我們將整個 mark 函數(shù)簡化為:

runtime.mark = function(genFun) {
  var generator = Object.create({
    next: function(arg) {
      return this._invoke("next", arg)
    }
  });
  genFun.prototype = generator;
  return genFun;
};
wrap 函數(shù)

除了設(shè)置關(guān)系鏈之外,mark 函數(shù)的返回值 genFun 還作為了 wrap 函數(shù)的第二個參數(shù)傳入:

function helloWorldGenerator() {
  return regeneratorRuntime.wrap(
    function helloWorldGenerator$(_context) {
      ...
    },
    _marked,
    this
  );
}

我們再看下 wrap 函數(shù):

function wrap(innerFn, outerFn, self) {
  var generator = Object.create(outerFn.prototype);
  var context = new Context([]);
  generator._invoke = makeInvokeMethod(innerFn, self, context);

  return generator;
}

所以當執(zhí)行 var hw = helloWorldGenerator(); 的時候,其實執(zhí)行的是 wrap 函數(shù),wrap 函數(shù)返回了 generator,generator 是一個對象,原型是 outerFn.prototype, outerFn.prototype 其實就是 genFun.prototype, genFun.prototype 是一個空對象,原型上有 next() 方法。

所以當你執(zhí)行 hw.next() 的時候,執(zhí)行的其實是 hw 原型的原型上的 next 函數(shù),next 函數(shù)執(zhí)行的又是 hw 的 _invoke 函數(shù):

generator._invoke = makeInvokeMethod(innerFn, self, context);

innerFn 就是 wrap 包裹的那個函數(shù),其實就是 helloWordGenerato$ 函數(shù),吶,就是這個函數(shù):

function helloWorldGenerator$(_context) {
  while (1) {
    switch ((_context.prev = _context.next)) {
      case 0:
        _context.next = 2;
        return "hello";

      case 2:
        _context.next = 4;
        return "world";

      case 4:
        return _context.abrupt("return", "ending");

      case 5:
      case "end":
        return _context.stop();
    }
  }
}

而 context 你可以直接理解為這樣一個全局對象:

var ContinueSentinel = {};

var context = {
  done: false,
  method: "next",
  next: 0,
  prev: 0,
  abrupt: function(type, arg) {
    var record = {};
    record.type = type;
    record.arg = arg;

    return this.complete(record);
  },
  complete: function(record, afterLoc) {
    if (record.type === "return") {
      this.rval = this.arg = record.arg;
      this.method = "return";
      this.next = "end";
    }

    return ContinueSentinel;
  },
  stop: function() {
    this.done = true;
    return this.rval;
  }
};

每次 hw.next 的時候,就會修改 next 和 prev 屬性的值,當在 generator 函數(shù)中 return 的時候會執(zhí)行 abrupt,abrupt 中又會執(zhí)行 complete,執(zhí)行完 complete,因為 this.next = end 的緣故,再執(zhí)行就會執(zhí)行 stop 函數(shù)。

我們來看下 makeInvokeMethod 函數(shù):

var ContinueSentinel = {};

function makeInvokeMethod(innerFn, self, context) {
  var state = "start";

  return function invoke(method, arg) {

    if (state === "completed") {
      return { value: undefined, done: true };
    }

    context.method = method;
    context.arg = arg;

    while (true) {

      state = "executing";

      var record = {
        type: "normal",
        arg: innerFn.call(self, context)
      };
      if (record.type === "normal") {

        state = context.done
          ? "completed"
          : "yield";

        if (record.arg === ContinueSentinel) {
          continue;
        }

        return {
          value: record.arg,
          done: context.done
        };

      }
    }
  };
}

基本的執(zhí)行過程就不分析了,我們重點看第三次執(zhí)行 hw.next() 的時候:

第三次執(zhí)行 hw.next() 的時候,其實執(zhí)行了

this._invoke("next", undefined);

我們在 invoke 函數(shù)中構(gòu)建了一個 record 對象:

var record = {
  type: "normal",
  arg: innerFn.call(self, context)
};

而在 innerFn.call(self, context) 中,因為 _context.next 為 4 的緣故,其實執(zhí)行了:

_context.abrupt("return", "ending");

而在 abrupt 中,我們又構(gòu)建了一個 record 對象:

var record = {};
record.type = "return";
record.arg = "ending";

然后執(zhí)行了 this.complete(record),

在 complete 中,因為 record.type === "return"

this.rval = "ending";
this.method = "return";
this.next = "end";

然后返回了全局對象 ContinueSentinel,其實就是一個全局空對象。

然后在 invoke 函數(shù)中,因為 record.arg === ContinueSentinel 的緣故,沒有執(zhí)行后面的 return 語句,就直接進入下一個循環(huán)。

于是又執(zhí)行了一遍 innerFn.call(self, context),此時 _context.next 為 end, 執(zhí)行了 _context.stop(), 在 stop 函數(shù)中:

this.done = true;
return this.rval; // this.rval 其實就是 `ending`

所以最終返回的值為:

{
  value: "ending",
  done: true
};

之后,我們再執(zhí)行 hw.next() 的時候,因為 state 已經(jīng)是 "completed" 的緣故,直接就返回 { value: undefined, done: true}

不完整但可用的源碼

當然這個過程,看文字理解起來可能有些難度,不完整但可用的代碼如下,你可以斷點調(diào)試查看具體的過程:

(function() {
  var ContinueSentinel = {};

  var mark = function(genFun) {
    var generator = Object.create({
      next: function(arg) {
        return this._invoke("next", arg);
      }
    });
    genFun.prototype = generator;
    return genFun;
  };

  function wrap(innerFn, outerFn, self) {
    var generator = Object.create(outerFn.prototype);

    var context = {
      done: false,
      method: "next",
      next: 0,
      prev: 0,
      abrupt: function(type, arg) {
        var record = {};
        record.type = type;
        record.arg = arg;

        return this.complete(record);
      },
      complete: function(record, afterLoc) {
        if (record.type === "return") {
          this.rval = this.arg = record.arg;
          this.method = "return";
          this.next = "end";
        }

        return ContinueSentinel;
      },
      stop: function() {
        this.done = true;
        return this.rval;
      }
    };

    generator._invoke = makeInvokeMethod(innerFn, context);

    return generator;
  }

  function makeInvokeMethod(innerFn, context) {
    var state = "start";

    return function invoke(method, arg) {
      if (state === "completed") {
        return { value: undefined, done: true };
      }

      context.method = method;
      context.arg = arg;

      while (true) {
        state = "executing";

        var record = {
          type: "normal",
          arg: innerFn.call(self, context)
        };

        if (record.type === "normal") {
          state = context.done ? "completed" : "yield";

          if (record.arg === ContinueSentinel) {
            continue;
          }

          return {
            value: record.arg,
            done: context.done
          };
        }
      }
    };
  }

  window.regeneratorRuntime = {};

  regeneratorRuntime.wrap = wrap;
  regeneratorRuntime.mark = mark;
})();

var _marked = regeneratorRuntime.mark(helloWorldGenerator);

function helloWorldGenerator() {
  return regeneratorRuntime.wrap(
    function helloWorldGenerator$(_context) {
      while (1) {
        switch ((_context.prev = _context.next)) {
          case 0:
            _context.next = 2;
            return "hello";

          case 2:
            _context.next = 4;
            return "world";

          case 4:
            return _context.abrupt("return", "ending");

          case 5:
          case "end":
            return _context.stop();
        }
      }
    },
    _marked,
    this
  );
}

var hw = helloWorldGenerator();

console.log(hw.next());
console.log(hw.next());
console.log(hw.next());
console.log(hw.next());
ES6 系列

ES6 系列目錄地址:https://github.com/mqyqingfeng/Blog

ES6 系列預計寫二十篇左右,旨在加深 ES6 部分知識點的理解,重點講解塊級作用域、標簽模板、箭頭函數(shù)、Symbol、Set、Map 以及 Promise 的模擬實現(xiàn)、模塊加載方案、異步處理等內(nèi)容。

如果有錯誤或者不嚴謹?shù)牡胤?,請?wù)必給予指正,十分感謝。如果喜歡或者有所啟發(fā),歡迎 star,對作者也是一種鼓勵。

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/98858.html

相關(guān)文章

  • ES6 系列 Babel Async 編譯成了什么樣子

    摘要:大約后輸出我們直接在官網(wǎng)的粘貼上述代碼,然后查看代碼編譯成什么樣子相關(guān)的代碼我們在系列之將編譯成了什么樣子中已經(jīng)介紹過了,這次我們重點來看看函數(shù)以上這段代碼主要是用來實現(xiàn)的自動執(zhí)行以及返回。 前言 本文就是簡單介紹下 Async 語法編譯后的代碼。 Async const fetchData = (data) => new Promise((resolve) => setTimeout...

    wangym 評論0 收藏0
  • ES6 走馬觀花(ECMAScript2015 新特性)

    摘要:字面上是生成器的意思,在里是迭代器生成器,用于生成一個迭代器對象。當執(zhí)行的時候,并不執(zhí)行函數(shù)體,而是返回一個迭代器。迭代器具有方法,每次調(diào)用方法,函數(shù)就執(zhí)行到語句的地方。也有觀點極力反對,認為隱藏了本身原型鏈的語言特性,使其更難理解。 本文為 ES6 系列的第一篇。旨在給新同學一些指引,帶大家走近 ES6 新特性。簡要介紹: 什么是 ES6 它有哪些明星特性 它可以運行在哪些環(huán)境 ...

    wangzy2019 評論0 收藏0
  • 【React進階系列】從零開始手把手教你實現(xiàn)一個Virtual DOM(二)

    摘要:上集回顧從零開始手把手教你實現(xiàn)一個一上一集我們介紹了什么是,為什么要用,以及我們要怎樣來實現(xiàn)一個。完成后,在命令行中輸入安裝下依賴。最后返回這個目標節(jié)點。明天,我們迎接挑戰(zhàn),開始處理數(shù)據(jù)變動引起的重新渲染,我們要如何新舊,生成補丁,修改。 上集回顧 從零開始手把手教你實現(xiàn)一個Virtual DOM(一)上一集我們介紹了什么是VDOM,為什么要用VDOM,以及我們要怎樣來實現(xiàn)一個VDOM...

    dendoink 評論0 收藏0
  • ES6 系列 let 和 const

    摘要:塊級作用域存在于函數(shù)內(nèi)部塊中字符和之間的區(qū)域和塊級聲明用于聲明在指定塊的作用域之外無法訪問的變量。和都是塊級聲明的一種。值得一提的是聲明不允許修改綁定,但允許修改值。這意味著當用聲明對象時沒有問題報錯臨時死區(qū)臨時死區(qū),簡寫為。 塊級作用域的出現(xiàn) 通過 var 聲明的變量存在變量提升的特性: if (condition) { var value = 1; } console.lo...

    PascalXie 評論0 收藏0
  • ES6系列 let 和 const

    摘要:聲明的變量不得改變值,這意味著,一旦聲明變量,就必須立即初始化,不能留到以后賦值。這在語法上,稱為暫時性死區(qū),簡稱。這表明函數(shù)內(nèi)部的變量與循環(huán)變量不在同一個作用域,有各自單獨的作用域。系列文章系列文章地址 showImg(https://segmentfault.com/img/bVbrjjC); 為什么需要塊級作用域 ES5 只有全局作用域和函數(shù)作用域,沒有塊級作用域,這帶來很多不合...

    libxd 評論0 收藏0

發(fā)表評論

0條評論

EddieChan

|高級講師

TA的文章

閱讀更多
最新活動
閱讀需要支付1元查看
<