Nodejs异步回调的优雅处理方法


Posted in NodeJs onSeptember 25, 2014

前言

Nodejs最大的亮点就在于事件驱动, 非阻塞I/O 模型,这使得Nodejs具有很强的并发处理能力,非常适合编写网络应用。在Nodejs中大部分的I/O操作几乎都是异步的,也就是我们处理I/O的操作结果基本上都需要在回调函数中处理,比如下面的这个读取文件内容的函数:

fs.readFile('/etc/passwd', function (err, data) {

  if (err) throw err;

  console.log(data);

});

那,我们读取两个文件,将这两个文件的内容合并到一起处理怎么办呢?大多数接触js不久的人可能会这么干:

fs.readFile('/etc/passwd', function (err, data) {

  if (err) throw err;

  fs.readFile('/etc/passwd2', function (err, data2) {

    if (err) throw err;

    // 在这里处理data和data2的数据

  });

});

那要是处理多个类似的场景,岂不是回调函数一层层的嵌套啊,这就是大家常说的回调金字塔或回调地狱(http://callbackhell.com/)的问题,也是让js小白最为头疼的问题。

这种层层嵌套的代码给开发带来了很多问题,主要体现在:

1.代码可能性变差
2.调试困难
3.出现异常后难以排查

本文主要是介绍如何优雅的处理以上异步回调问题。

初级方案:通过递归处理异步回调

我们可以使用递归作为代码的执行控制工具。把需要执行的操作封装到一个函数中,在回调函数中通过递归调用控制代码的执行流程,废话不多说,上个代码吧:

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) {

    // 这里处理文件数据

    parseFile();  // 处理完毕后,通过递归调用处理下一个文件

  });

}
// 开始处理

parseFile();

以上代码已依次处理数组中的文件为例,介绍了通过递归的方式控制代码的执行流程。

应用到一些简单的场景中还是不错的,比如:我们将一个数组中的数据,依次保存到数据库中就可以采用这种方式。

通过递归的方式可以解决一些简单的异步回调问题。不过对于处理复杂的异步回调还是显得有些无能为力(如需要同步多个异步操作的结果)。

华丽点:采用Async、Q、Promise等第三方库处理异步回调

为了更好的处理嵌套回调的问题,可以考虑采用一些第三方专门处理异步的库,当然有能力的完全可以自己写个异步处理的辅助工具。

比较常用的处理异步的库有:async,q还有promise。从npmjs.org网站上来看,async的火热程度最高。以前用过async,确实也挺方便的,各种异步处理的控制流实现的也挺好。

我们将最初的同时读取两个文件的代码使用async处理下,示例如下:

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的数据,每个文件的内容从results中获取

});

通过async模块,可以很好的控制异步的执行流程了,也算是解决了层层回调的问题,代码比以前算是清晰了些,不过依旧还是离不开回调函数。

想想如果能够在不使用回调函数的情况下,处理异步,岂不是很爽,接下来,我们谈谈使用ES6的新特性来实现这一目标。

优雅点:拥抱ES6,替代回调函数,解决回调地狱问题

话说EcmaScript Harmony (ES6)给js引入了不少新特性,对ES6不太了解的同学,可以自行百度一下。

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

本文介绍的是使用Generator特性替代回调函数,对Generator不了解?可以看看这里。

这里用到了co和thunkify两个模块,大家使用npm install命令安装之。

还是以本文刚开始提到的问题为例,使用generator特性的实例代码如下:

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);

})();

处理代码中的异常也是很简单的,只需要这样就OK了:

try {

  var test1 = yield readFile('test1.txt');

} catch (e) {

  // 在这里处理异常

}

这种代码是不是优雅很多了?像写同步代码一样处理异步,是不是很爽!

nodejs领域中进行Web开发,最火的框架莫过于express了,值得一提的是express的核心成员TJ大神有领导了一个新的Web框架——koa,宣称是下一代的Web开发框架,koa真是借助了ES6的generator这一特性,让我们在开发Web系统的时候避免陷入层层的回调用。

总结

引用一下fibjs项目宣传的一句话:Less Callback, More Girls - 更少回调, 更多妹子

NodeJs 相关文章推荐
Nodejs实现多人同时在线移动鼠标的小游戏分享
Dec 06 NodeJs
轻松创建nodejs服务器(7):阻塞操作的实现
Dec 18 NodeJs
nodejs加密Crypto的实例代码
Jul 07 NodeJs
nodejs模块nodemailer基本使用-邮件发送示例(支持附件)
Mar 28 NodeJs
让nodeJS支持ES6的词法----babel的安装和使用方法
Jul 31 NodeJs
详解IWinter 一个路由转控制器的 Nodejs 库
Nov 15 NodeJs
nodejs+mongodb aggregate级联查询操作示例
Mar 17 NodeJs
Nodejs异步回调之异常处理实例分析
Jun 22 NodeJs
Nodejs Express 通过log4js写日志到Logstash(ELK)
Aug 30 NodeJs
NodeJs crypto加密制作token的实现代码
Nov 15 NodeJs
如何利用nodejs实现命令行游戏
Nov 24 NodeJs
Nodejs 微信小程序消息推送的实现
Jan 20 NodeJs
nodejs命令行参数处理模块commander使用实例
Sep 17 #NodeJs
nodejs npm package.json中文文档
Sep 04 #NodeJs
使用Nodejs开发微信公众号后台服务实例
Sep 03 #NodeJs
Nodejs+express+html5 实现拖拽上传
Aug 08 #NodeJs
如何正确使用Nodejs 的 c++ module 链接到 OpenSSL
Aug 03 #NodeJs
NodeJS学习笔记之网络编程
Aug 03 #NodeJs
基于 Docker 开发 NodeJS 应用
Jul 30 #NodeJs
You might like
如何去掉文章里的 html 语法
2006/10/09 PHP
php getimagesize 上传图片的长度和宽度检测代码
2010/05/15 PHP
php多个字符串替换成同一个的解决方法
2013/06/18 PHP
几行代码轻松实现PHP文件打包下载zip
2017/03/01 PHP
基于JavaScript 声明全局变量的三种方式详解
2013/05/07 Javascript
js实现图片漂浮效果的方法
2015/03/02 Javascript
js改变embed标签src值的方法
2015/04/10 Javascript
使用Browserify配合jQuery进行编程的超级指南
2015/07/28 Javascript
JS提交form表单实例分析
2015/12/10 Javascript
jquery选择器中的空格与大于号>、加号+与波浪号~的区别介绍
2016/06/24 Javascript
JS实现兼容火狐及IE iframe onload属性的遮罩层隐藏及显示效果
2016/08/23 Javascript
bootstrap导航栏、下拉菜单、表单的简单应用实例解析
2017/01/06 Javascript
jQuery实现二维码扫描功能
2017/01/09 Javascript
微信小程序开发之相册选择和拍照详解及实例代码
2017/02/22 Javascript
使用layui实现的左侧菜单栏以及动态操作tab项方法
2019/09/10 Javascript
前端深入理解Typescript泛型概念
2020/03/09 Javascript
vue抽出组件并传值实例
2020/07/31 Javascript
[00:36]TI7不朽珍藏III——斯温不朽展示
2017/07/15 DOTA
[05:40]DOTA2荣耀之路6:Wings最后进攻
2018/05/30 DOTA
python实现dnspod自动更新dns解析的方法
2014/02/14 Python
Python中的模块和包概念介绍
2015/04/13 Python
对python GUI实现完美进度条的示例详解
2018/12/13 Python
python 对类的成员函数开启线程的方法
2019/01/22 Python
Python3 修改默认环境的方法
2019/02/16 Python
Python3.x+pyqtgraph实现数据可视化教程
2020/03/14 Python
python线性插值解析
2020/07/05 Python
Python jieba结巴分词原理及用法解析
2020/11/05 Python
python如何获得list或numpy数组中最大元素对应的索引
2020/11/16 Python
宝拉珍选官方旗舰店:2%水杨酸精华液,收缩毛孔粗大和祛痘
2018/07/01 全球购物
日本最大的购物网站:日本乐天市场(Rakuten Ichiba)
2020/11/04 全球购物
kfc实习自我鉴定
2013/12/14 职场文书
个人求职信范文分享
2014/01/06 职场文书
会计做账心得体会
2016/01/22 职场文书
IDEA使用SpringAssistant插件创建SpringCloud项目
2021/06/23 Java/Android
Spring Boot两种全局配置和两种注解的操作方法
2021/06/29 Java/Android
python神经网络学习 使用Keras进行简单分类
2022/05/04 Python