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

資訊專欄INFORMATION COLUMN

co源碼分析及其實(shí)踐

vincent_xyb / 1028人閱讀

摘要:返回的結(jié)果是一個(gè)對象,類似于表示本次后面執(zhí)行之后返回的結(jié)果。對象用于一個(gè)異步操作的最終完成或失敗及其結(jié)果值的表示簡單點(diǎn)說就是處理異步請求。源碼分析主要脈絡(luò)函數(shù)調(diào)用后,返回一個(gè)實(shí)例。參考鏈接解釋對象的用法的源碼及其用法

本文始發(fā)于我的個(gè)人博客,如需轉(zhuǎn)載請注明出處。
為了更好的閱讀體驗(yàn),可以直接進(jìn)去我的個(gè)人博客看。

前言 知識(shí)儲(chǔ)備

閱讀本文需要對GeneratorPromise有一個(gè)基本的了解。

這里我簡單地介紹一下兩者的用法。

Generator

關(guān)于Generator的用法,推薦MDN上面的解釋function *函數(shù),里面非常詳細(xì)。

用一句話總結(jié)就是,generator函數(shù)是回調(diào)地獄的一種解決方案,它跟promise類似,但是卻可以以同步的方式來書寫代碼,而避免了promise的鏈?zhǔn)秸{(diào)用。

它的執(zhí)行過程在于調(diào)用生成器函數(shù)(generator function)后,會(huì)返回一個(gè)iterator(迭代)對象,即Generator對象,但是它并不會(huì)立刻執(zhí)行里面的代碼。

它有幾個(gè)方法,next(), throw()return()。調(diào)用next()方法后,它會(huì)找到第一個(gè)yield關(guān)鍵字(直到找到程序底部或者return語句),每次程序運(yùn)行到y(tǒng)ield關(guān)鍵字時(shí),程序便會(huì)暫停,保存當(dāng)前環(huán)境里面的變量的值,然后可以跳出當(dāng)前運(yùn)行環(huán)境去執(zhí)行yield后面的代碼,再把結(jié)果返回回來。

返回的結(jié)果是一個(gè)對象,類似于{value: "", done: false}, value表示本次yield后面執(zhí)行之后返回的結(jié)果。如果是Promise實(shí)例,則是返回resolved后的值。done表示迭代器是否執(zhí)行完畢,若為true,則表示當(dāng)前生成器函數(shù)已經(jīng)產(chǎn)生了最后輸出的值,即生成器函數(shù)已經(jīng)返回。

下面是一個(gè)簡單的例子:

const gen = function *() {
  let index = 0;
  while(index < 3)
    yield index++;
  return "All done."
};

const g = gen();
console.log(g.constructor); // output: GeneratorFunction {}
console.log(g.next()); // output: { value: 0, done: false }
console.log(g.next()); // output: { value: 1, done: false }
console.log(g.next()); // output: { value: 2, done: false }
console.log(g.next()); // output: { value: "All done.", done: true }
console.log(g.next()); // output: { value: undefined, done: true }
Promise

關(guān)于Promise的用法,可以查閱我之前寫過的一篇文章《關(guān)于ES6中Promise的用法》,寫得比較詳細(xì)。

Promise對象用于一個(gè)異步操作的最終完成(或失?。┘捌浣Y(jié)果值的表示(簡單點(diǎn)說就是處理異步請求)。Promise核心就在于里面狀態(tài)的變換,是rejectedresolved還是pending,還有就是原型鏈上的then()方法,它可以傳遞本次狀態(tài)轉(zhuǎn)換后返回的值。

進(jìn)入主題

由于實(shí)際需要,這幾天學(xué)習(xí)了koa2.x框架,但是它已經(jīng)不推薦使用generator函數(shù)了,推薦用async/await組合。

koa2.x的最新用法:

async/await(node v7.6+):

const Koa = require("koa");
const app = new Koa();

app.use(async (ctx, next) => {
  const start = Date.now();
  await next();
  const ms = Date.now() - start;
  console.log(`${ctx.method} ${ctx.url} - ${ms}ms`);
});

common 用法:

const Koa = require("koa");
const app = new Koa();

// response
app.use(ctx => {
  ctx.body = "Hello Koa";
});

app.listen(3000);

由于本地的Node版本是v6.11.5,而使用async/await則需要Node版本v7.6以上,所以我想有沒有什么模塊能夠把koa2.x版本的語法兼容koa1.x的語法。koa1.x語法的關(guān)鍵在于generator/yield組合。通過yield可以很方便地暫停程序的執(zhí)行,并改變執(zhí)行環(huán)境。

這時(shí)候我找到了TJ大神寫的co模塊,它可以讓異步流程同步化,還有koa-convert模塊等等,這里著重介紹co模塊。

co在koa2.x里面的用法如下:

const Koa = require("koa");
const app = new Koa();
const co = require("co");

// response
app.use(co.wrap(function *(ctx, next) {
  yield next();
  // yield someAyncOperation;
  //  ... 
  ctx.body = "co";
}));

app.listen(3000);

co模塊不僅可以配合koa框架充當(dāng)中間件的轉(zhuǎn)換函數(shù)使用,還支持批量執(zhí)行g(shù)enerator函數(shù),這樣就無需手動(dòng)調(diào)用多次next()來獲取結(jié)果了。

它支持的參數(shù)有函數(shù)、promise、generator、數(shù)組和對象

// co的源碼
return onRejected(new TypeError("You may only yield a function, promise, generator, array, or object, "
        + "but the following object was passed: "" + String(ret.value) + """));

下面舉一個(gè)co傳遞進(jìn)來一個(gè)generator函數(shù)的例子:

// 這里模擬一個(gè)generator函數(shù)調(diào)用
const co = require("co");

co(gen).then(data => {
  // output: then: ALL Done.
  console.log("then: " + data);
});

function *gen() {
  let data1 = yield pro1();
  // output: pro1 had resolved, data1 = I am promise1
  console.log("pro1 had resolved, data1 = " + data1);

  let data2 = yield pro2();
  // output: pro2 had resolved, data2 = I am promise2
  console.log("pro2 had resolved, data2 = " + data2);

  return "ALL Done."
}

function pro1() {
  return new Promise((resolve, reject) => {
    setTimeout(resolve, 2000, "I am promise1");
  });
}

function pro2() {
  return new Promise((resolve, reject) => {
    setTimeout(resolve, 1000, "I am promise2");
  });
}

我覺得co()函數(shù)很神奇,里面究竟經(jīng)過了什么樣的轉(zhuǎn)換?抱著一顆好奇心,讀了一下co的源碼。

co源碼分析 主要脈絡(luò)

co函數(shù)調(diào)用后,返回一個(gè)Promise實(shí)例。

co的思想就是將一個(gè)傳遞進(jìn)來的參數(shù)進(jìn)行合法化,再通過轉(zhuǎn)換成Promise實(shí)例返回出去。如果參數(shù)fn是generator函數(shù)的話,里面還可以自動(dòng)進(jìn)行遍歷,執(zhí)行g(shù)enerator函數(shù)里面的yield關(guān)鍵字后面的內(nèi)容,并返回結(jié)果,也就是不斷地調(diào)用fn().next()方法,再通過傳遞返回的Promise實(shí)例resolved后的值,從而達(dá)到同步執(zhí)行g(shù)enerator函數(shù)的效果。

這里要注意,co里面最主要的是要理解Promise實(shí)例和Generator對象,它們是co函數(shù)里面的程序自動(dòng)遍歷執(zhí)行的關(guān)鍵

下面解釋一下co模塊里面的最重要的兩部分,一個(gè)是generator函數(shù)的自動(dòng)調(diào)用,另外一個(gè)是參數(shù)的Promise化。

第一,generator函數(shù)的自動(dòng)調(diào)用(中文部分是我的解釋):

function co(gen) {
  // 保存當(dāng)前的執(zhí)行環(huán)境
  var ctx = this;
  // 切割出函數(shù)調(diào)用時(shí)傳遞的參數(shù)
  var args = slice.call(arguments, 1)

  // we wrap everything in a promise to avoid promise chaining,
  // which leads to memory leak errors.
  // see https://github.com/tj/co/issues/180
  
  // 返回一個(gè)Promise實(shí)例
  return new Promise(function(resolve, reject) {
    // 如果gen是一個(gè)函數(shù),則返回一個(gè)新的gen函數(shù)的副本,
    // 里面綁定了this的指向,即ctx
    if (typeof gen === "function") gen = gen.apply(ctx, args);
    
    // 如果gen不存在或者gen.next不是一個(gè)函數(shù)
    // 就說明gen已經(jīng)調(diào)用完成,
    // 那么直接可以resolve(gen),返回Promise
    if (!gen || typeof gen.next !== "function") return resolve(gen);

    // 首次調(diào)用gen.next()函數(shù),假如存在的話
    onFulfilled();

    /**
     * @param {Mixed} res
     * @return {Promise}
     * @api private
     */

    function onFulfilled(res) {
      var ret;
      try {
        // 嘗試著獲取下一個(gè)yield后面代碼執(zhí)行后返回的值
        ret = gen.next(res);
      } catch (e) {
        return reject(e);
      }
      // 處理結(jié)果
      next(ret);
    }

    /**
     * @param {Error} err
     * @return {Promise}
     * @api private
     */

    function onRejected(err) {
      var ret;
      try {
        // 嘗試拋出錯(cuò)誤
        ret = gen.throw(err);
      } catch (e) {
        return reject(e);
      }
      // 處理結(jié)果
      next(ret);
    }

    /**
     * Get the next value in the generator,
     * return a promise.
     *
     * @param {Object} ret
     * @return {Promise}
     * @api private
     */

    // 這個(gè)next()函數(shù)是最為關(guān)鍵的一部分,
    // 里面幾乎包含了generator自動(dòng)調(diào)用實(shí)現(xiàn)的核心
    function next(ret) {
      // 如果ret.done === true, 
      // 證明generator函數(shù)已經(jīng)執(zhí)行完畢
      // 即已經(jīng)返回了值
      if (ret.done) return resolve(ret.value);
      
      // 把ret.value轉(zhuǎn)換成Promise對象繼續(xù)調(diào)用
      var value = toPromise.call(ctx, ret.value);
      
      // 如果存在,則把控制權(quán)交給onFulfilled和onRejected,
      // 實(shí)現(xiàn)遞歸調(diào)用
      if (value && isPromise(value)) return value.then(onFulfilled, onRejected);
      
      // 否則最后直接拋出錯(cuò)誤
      return onRejected(new TypeError("You may only yield a function, promise, generator, array, or object, "
        + "but the following object was passed: "" + String(ret.value) + """));
    }
  });
}

對于以上代碼中的onFulfilledonRejected,我們可以把它們看成是co模塊對于resolvereject封裝的加強(qiáng)版。

第二,參數(shù)Promise化,我們來看一下co中的toPromise的實(shí)現(xiàn):

function toPromise(obj) {
  if (!obj) return obj;
  if (isPromise(obj)) return obj;
  if (isGeneratorFunction(obj) || isGenerator(obj)) return co.call(this, obj);
  if ("function" == typeof obj) return thunkToPromise.call(this, obj);
  if (Array.isArray(obj)) return arrayToPromise.call(this, obj);
  if (isObject(obj)) return objectToPromise.call(this, obj);
  return obj;
}

toPromise的本質(zhì)上就是通過判定參數(shù)的類型,然后再通過轉(zhuǎn)移控制權(quán)給不同的參數(shù)處理函數(shù),從而獲取到期望返回的值。

關(guān)于參數(shù)的類型的判斷,看一下源碼就能理解了,比較簡單。

我們著重來分析一下objectToPromise的實(shí)現(xiàn):

function objectToPromise(obj){
  // 獲取一個(gè)和傳入的對象一樣構(gòu)造器的對象
  var results = new obj.constructor();
  
  // 獲取對象的所有可以遍歷的key
  var keys = Object.keys(obj);
  var promises = [];
  for (var i = 0; i < keys.length; i++) {
    var key = keys[i];
    
    // 對于數(shù)組的每一個(gè)項(xiàng)都調(diào)用一次toPromise方法,變成Promise對象
    var promise = toPromise.call(this, obj[key]);
    
    // 如果里面是Promise對象的話,則取出e里面resolved后的值
    if (promise && isPromise(promise)) defer(promise, key);
    else results[key] = obj[key];
  }
  
  // 并行,按順序返回結(jié)果,返回一個(gè)數(shù)組
  return Promise.all(promises).then(function () {
    return results;
  });

  // 根據(jù)key來獲取Promise實(shí)例resolved后的結(jié)果,
  // 從而push進(jìn)結(jié)果數(shù)組results中
  function defer(promise, key) {
    // predefine the key in the result
    results[key] = undefined;
    promises.push(promise.then(function (res) {
      results[key] = res;
    }));
  }
}

上面理解的關(guān)鍵就在于把key遍歷,如果key對應(yīng)的value也是Promise對象的話,那么調(diào)用defer()方法來獲取resolved后的值。

編寫自己的generator函數(shù)運(yùn)行器

通過以上的簡單介紹,我們就可以嘗試來寫一個(gè)屬于自己的generator函數(shù)運(yùn)行器了,目標(biāo)功能是能夠自動(dòng)運(yùn)行function*函數(shù),并且里面的yield子句后面跟著的都是Promise實(shí)例

具體代碼(my-co.js)如下:

// my-co.js
module.exports = my-co;

let my-co = function (gen) {
  // gen是一個(gè)具有Promise的生成器函數(shù)
  const g = gen(); // 迭代器
  
  // 首次調(diào)用next
  next();

  function next(val) {
    let ret = g.next(val); // 調(diào)用ret
    if (ret.done) {
      return ret.value;
    }

    if (ret && "function" === typeof ret.value.then) {
      ret.value.then( (data) => {
        // 繼續(xù)循環(huán)下去
        return next(data); // promise resolved
      });
    }
  }
};

這樣我們就可以在test.js文件中調(diào)用了:

// test.js
const myCo = require("./my-co");
const fs = require("fs");

let gen = function *() {
  let data1 = yield pro1();
  console.log("data1: " + data1);

  let data2 = yield pro2();
  console.log("data2: " + data2);

  let data3 = yield pro3();
  console.log("data3: " + data3);

  let data4 = yield pro4(data1 + "
" + data2 + "
" + data3);
  console.log("data4: " + data4);

  return "All done."
};

// 調(diào)用myCo
myCo(gen);

// 延遲兩秒resolve
function pro1() {
  return new Promise((resolve, reject) => {
    setTimeout(resolve, 2000, "promise1 resolved");
  });
}

// 延遲一秒resolve
function pro2() {
  return new Promise((resolve, reject) => {
    setTimeout(resolve, 1000, "promise2 resolved");
  });
}

// 寫入Hello World到./1.txt文件中
function pro3() {
  return new Promise((resolve, reject) => {
    fs.appendFile("./1.txt", "Hello World
", function(err) {
      resolve("write-1 success");
    });
  });
}

// 寫入content到./1.txt文件中
function pro4(content) {
  return new Promise((resolve, reject) => {
    fs.appendFile("./1.txt", content, function(err) {
      resolve("write-2 success");
    });
  });
}

控制臺(tái)輸出結(jié)果:

// output
data1: promise1 resolved
data2: promise2 resolved
data3: write-1 success
data4: write-2 success

./1.txt文件內(nèi)容:

Hello World
promise1 resolved
promise2 resolved
write-1 success

由上可知,運(yùn)行的結(jié)果符合我們的期望。

雖然這個(gè)運(yùn)行器很簡單,后面只支持Promise實(shí)例,并且也不支持多種參數(shù),但是卻引導(dǎo)出了一個(gè)思路,促使我們思考怎么去展示我們的代碼,還有就是很有效地避免了多重then,以同步的方式來書寫異步代碼。Promise解決的是回調(diào)地獄的問題(callback hell),而Generator解決的是代碼的書寫方式。孰優(yōu)孰劣,全在于個(gè)人意愿。

總結(jié)

以上分析了co部分源碼的精髓,講到了co函數(shù)里面generator函數(shù)自動(dòng)遍歷執(zhí)行的機(jī)制,還講到了co里面最為關(guān)鍵的objectToPromise()方法。

在文章的后面我們編寫了一個(gè)屬于自己的generator函數(shù)遍歷器,其中主要的是next()方法,它可以檢測我們yield后面Promise操作是否完成。如果generator的狀態(tài)done還沒有置為true,那么繼續(xù)調(diào)用next(val)方法,并把上一次yield操作獲取到的值傳遞下去。

有時(shí)候在引用別人的模塊出現(xiàn)問題時(shí),如果在網(wǎng)上找不到自己期望的答案,那么我們可以根據(jù)自己的能力來選擇性地分析一下作者的源碼,看源碼是一種很好的成長方式。

坦白說,這是我第一次深入分析模塊的源碼,co模塊的源碼包括注釋和空行只有230多行左右,所以這是一個(gè)很好的切入點(diǎn)。里面代碼雖少,但是理解卻不易。

如果以上所述有什么問題,歡迎反饋。

感謝支持。

參考鏈接

MDN - Promise解釋

MDN - Generator對象的用法

TJ - co的源碼及其用法

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

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

相關(guān)文章

  • 前端每周清單半年盤點(diǎn)之 JavaScript 篇

    摘要:前端每周清單專注前端領(lǐng)域內(nèi)容,以對外文資料的搜集為主,幫助開發(fā)者了解一周前端熱點(diǎn)分為新聞熱點(diǎn)開發(fā)教程工程實(shí)踐深度閱讀開源項(xiàng)目巔峰人生等欄目。背后的故事本文是對于年之間世界發(fā)生的大事件的詳細(xì)介紹,闡述了從提出到角力到流產(chǎn)的前世今生。 前端每周清單專注前端領(lǐng)域內(nèi)容,以對外文資料的搜集為主,幫助開發(fā)者了解一周前端熱點(diǎn);分為新聞熱點(diǎn)、開發(fā)教程、工程實(shí)踐、深度閱讀、開源項(xiàng)目、巔峰人生等欄目。歡迎...

    Vixb 評(píng)論0 收藏0
  • 前端每周清單半年盤點(diǎn)之 React 與 ReactNative 篇

    摘要:前端每周清單半年盤點(diǎn)之與篇前端每周清單專注前端領(lǐng)域內(nèi)容,以對外文資料的搜集為主,幫助開發(fā)者了解一周前端熱點(diǎn)分為新聞熱點(diǎn)開發(fā)教程工程實(shí)踐深度閱讀開源項(xiàng)目巔峰人生等欄目。與求同存異近日,宣布將的構(gòu)建工具由遷移到,引發(fā)了很多開發(fā)者的討論。 前端每周清單半年盤點(diǎn)之 React 與 ReactNative 篇 前端每周清單專注前端領(lǐng)域內(nèi)容,以對外文資料的搜集為主,幫助開發(fā)者了解一周前端熱點(diǎn);分為...

    Barry_Ng 評(píng)論0 收藏0
  • 前端每周清單半年盤點(diǎn)之 Node.js 篇

    摘要:前端每周清單專注前端領(lǐng)域內(nèi)容,以對外文資料的搜集為主,幫助開發(fā)者了解一周前端熱點(diǎn)分為新聞熱點(diǎn)開發(fā)教程工程實(shí)踐深度閱讀開源項(xiàng)目巔峰人生等欄目。對該漏洞的綜合評(píng)級(jí)為高危。目前,相關(guān)利用方式已經(jīng)在互聯(lián)網(wǎng)上公開,近期出現(xiàn)攻擊嘗試爆發(fā)的可能。 前端每周清單專注前端領(lǐng)域內(nèi)容,以對外文資料的搜集為主,幫助開發(fā)者了解一周前端熱點(diǎn);分為新聞熱點(diǎn)、開發(fā)教程、工程實(shí)踐、深度閱讀、開源項(xiàng)目、巔峰人生等欄目。歡...

    kid143 評(píng)論0 收藏0
  • PHP小知識(shí)點(diǎn)

    摘要:那些瑣碎的知識(shí)點(diǎn)作者記錄的的很奇特很難記的知識(shí)點(diǎn)。易錯(cuò)知識(shí)點(diǎn)整理注意和的區(qū)別中和都是輸出的作用,但是兩者之間還是有細(xì)微的差別。今天手頭不忙,總結(jié)一下,分享過程中掌握的知識(shí)點(diǎn)。 深入理解 PHP 之:Nginx 與 FPM 的工作機(jī)制 這篇文章從 Nginx 與 FPM 的工作機(jī)制出發(fā),探討配置背后的原理,讓我們真正理解 Nginx 與 PHP 是如何協(xié)同工作的。 PHP 那些瑣碎的知識(shí)...

    hover_lew 評(píng)論0 收藏0
  • koa源碼分析-co模塊以及thunk

    摘要:以及模塊之前都是返回的函數(shù)之后的都是返回在語言中,函數(shù)替換的是將多參數(shù)函數(shù),替換成單參數(shù)的版本,且只接受回調(diào)函數(shù)作為參數(shù)。 Thunk以及CO模塊 co4.0之前都是返回的thunk函數(shù)之后的都是返回promise thunk thunk:在 JavaScript 語言中,Thunk 函數(shù)替換的是將多參數(shù)函數(shù),替換成單參數(shù)的版本,且只接受回調(diào)函數(shù)作為參數(shù)。 // 正常版本的readFi...

    caikeal 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<