小男孩‘自慰网亚洲一区二区,亚洲一级在线播放毛片,亚洲中文字幕av每天更新,黄aⅴ永久免费无码,91成人午夜在线精品,色网站免费在线观看,亚洲欧洲wwwww在线观看

分享

如何優(yōu)雅的處理Nodejs中的異步回調(diào)

 昵稱10504424 2014-09-25

前言

Nodejs最大的亮點(diǎn)就在于事件驅(qū)動(dòng), 非阻塞I/O 模型,這使得Nodejs具有很強(qiáng)的并發(fā)處理能力,非常適合編寫網(wǎng)絡(luò)應(yīng)用。在Nodejs中大部分的I/O操作幾乎都是異步的,也就是我們處理I/O的操作結(jié)果基本上都需要在回調(diào)函數(shù)中處理,比如下面的這個(gè)讀取文件內(nèi)容的函數(shù):

fs.readFile('/etc/passwd', function (err, data) {
  if (err) throw err;
  console.log(data);
});

那,我們讀取兩個(gè)文件,將這兩個(gè)文件的內(nèi)容合并到一起處理怎么辦呢?大多數(shù)接觸js不久的人可能會(huì)這么干:

復(fù)制代碼
fs.readFile('/etc/passwd', function (err, data) {
  if (err) throw err;
  fs.readFile('/etc/passwd2', function (err, data2) {
    if (err) throw err;
    // 在這里處理data和data2的數(shù)據(jù)
  });
});
復(fù)制代碼

那要是處理多個(gè)類似的場景,豈不是回調(diào)函數(shù)一層層的嵌套啊,這就是大家常說的回調(diào)金字塔或回調(diào)地獄(http:///)的問題,也是讓js小白最為頭疼的問題。

這種層層嵌套的代碼給開發(fā)帶來了很多問題,主要體現(xiàn)在:

  • 代碼可能性變差
  • 調(diào)試?yán)щy
  • 出現(xiàn)異常后難以排查

本文主要是介紹如何優(yōu)雅的處理以上異步回調(diào)問題。

初級(jí)方案:通過遞歸處理異步回調(diào)

我們可以使用遞歸作為代碼的執(zhí)行控制工具。把需要執(zhí)行的操作封裝到一個(gè)函數(shù)中,在回調(diào)函數(shù)中通過遞歸調(diào)用控制代碼的執(zhí)行流程,廢話不多說,上個(gè)代碼吧:

復(fù)制代碼
var fs = require('fs');
// 要處理的文件列表
var files = ['file1', 'file2', 'file3'];

function parseFile () {
  if (files.length == 0) {
    return;
  }
  var file = files.shift();
  fs.readFile(file, function (err, data) {
    // 這里處理文件數(shù)據(jù)
    parseFile();  // 處理完畢后,通過遞歸調(diào)用處理下一個(gè)文件
  });
}

// 開始處理
parseFile();
復(fù)制代碼

以上代碼已依次處理數(shù)組中的文件為例,介紹了通過遞歸的方式控制代碼的執(zhí)行流程。

應(yīng)用到一些簡單的場景中還是不錯(cuò)的,比如:我們將一個(gè)數(shù)組中的數(shù)據(jù),依次保存到數(shù)據(jù)庫中就可以采用這種方式。

通過遞歸的方式可以解決一些簡單的異步回調(diào)問題。不過對(duì)于處理復(fù)雜的異步回調(diào)還是顯得有些無能為力(如需要同步多個(gè)異步操作的結(jié)果)。

華麗點(diǎn):采用Async、Q、Promise等第三方庫處理異步回調(diào)

為了更好的處理嵌套回調(diào)的問題,可以考慮采用一些第三方專門處理異步的庫,當(dāng)然有能力的完全可以自己寫個(gè)異步處理的輔助工具。

比較常用的處理異步的庫有:async,q還有promise。從網(wǎng)站上來看,async的火熱程度最高。以前用過async,確實(shí)也挺方便的,各種異步處理的控制流實(shí)現(xiàn)的也挺好。

我們將最初的同時(shí)讀取兩個(gè)文件的代碼使用async處理下,示例如下:

復(fù)制代碼
var async = require('async')
  , fs = require('fs');

async.parallel([
  function(callback){
    fs.readFile('/etc/passwd', function (err, data) {
      if (err) callback(err);
      callback(null, data);
    });
  },
  function(callback){
    fs.readFile('/etc/passwd2', function (err, data2) {
      if (err) callback(err);
      callback(null, data2);
    });
  }
],
function(err, results){
  // 在這里處理data和data2的數(shù)據(jù),每個(gè)文件的內(nèi)容從results中獲取
});
復(fù)制代碼

通過async模塊,可以很好的控制異步的執(zhí)行流程了,也算是解決了層層回調(diào)的問題,代碼比以前算是清晰了些,不過依舊還是離不開回調(diào)函數(shù)。

想想如果能夠在不使用回調(diào)函數(shù)的情況下,處理異步,豈不是很爽,接下來,我們談?wù)勈褂肊S6的新特性來實(shí)現(xiàn)這一目標(biāo)。

優(yōu)雅點(diǎn):擁抱ES6,替代回調(diào)函數(shù),解決回調(diào)地獄問題

話說EcmaScript Harmony (ES6)給js引入了不少新特性,對(duì)ES6不太了解的同學(xué),可以自行百度一下。

在nodejs中使用ES6的新特性,需要用v0.11.x以上的版本才行。

本文介紹的是使用Generator特性替代回調(diào)函數(shù),對(duì)Generator不了解?可以看看這里。

這里用到了co和thunkify兩個(gè)模塊,大家使用npm install命令安裝之。

還是以本文剛開始提到的問題為例,使用generator特性的實(shí)例代碼如下:

復(fù)制代碼
var fs = require('fs')
  , co = require('co')
  , thunkify = require('thunkify');

var readFile = thunkify(fs.readFile);

co(function *() {
  var test1 = yield readFile('test1.txt');
  var test2 = yield readFile('test2.txt');
  var test = test1.toString() + test2.toString();
  console.log(test);
})();
復(fù)制代碼

處理代碼中的異常也是很簡單的,只需要這樣就OK了:

try {
  var test1 = yield readFile('test1.txt');
} catch (e) {
  // 在這里處理異常
}

這種代碼是不是優(yōu)雅很多了?像寫同步代碼一樣處理異步,是不是很爽!

nodejs領(lǐng)域中進(jìn)行Web開發(fā),最火的框架莫過于express了,值得一提的是express的核心成員TJ大神有領(lǐng)導(dǎo)了一個(gè)新的Web框架——koa,宣稱是下一代的Web開發(fā)框架,koa真是借助了ES6的generator這一特性,讓我們?cè)陂_發(fā)Web系統(tǒng)的時(shí)候避免陷入層層的回調(diào)用。

總結(jié)

引用一下fibjs項(xiàng)目宣傳的一句話:Less Callback, More Girls - 更少回調(diào), 更多妹子

參考資料

http://blog./2014/03/12/node-v0-11-12-unstable/

http:///765

http:///2013/06/16/what-are-generators/

http://blog./using-es6-generators-in-nodejs/

http://www./article/1687

http://www./article/62609

據(jù)說看到好文章不推薦的人,服務(wù)器容易宕機(jī)!
本文版權(quán)歸翟士丹(Stan Zhai)博客園共有,歡迎轉(zhuǎn)載,但未經(jīng)作者同意必須保留此段聲明,且在文章頁面明顯位置給出原文連接,否則保留追究法律責(zé)任的權(quán)利。

分類: Node.js

    本站是提供個(gè)人知識(shí)管理的網(wǎng)絡(luò)存儲(chǔ)空間,所有內(nèi)容均由用戶發(fā)布,不代表本站觀點(diǎn)。請(qǐng)注意甄別內(nèi)容中的聯(lián)系方式、誘導(dǎo)購買等信息,謹(jǐn)防詐騙。如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,請(qǐng)點(diǎn)擊一鍵舉報(bào)。
    轉(zhuǎn)藏 分享 獻(xiàn)花(0

    0條評(píng)論

    發(fā)表

    請(qǐng)遵守用戶 評(píng)論公約

    類似文章 更多