NodeJS异步流程控制简单介绍

有这样一个需求,用户注册的时候,判断用户名和邮箱是否已经被占用。

用户注册

传统的实现思路

  • 根据用户名查找记录,如果存在记录,证明用户名已被占用
  • 根据邮箱查找记录,如果存在记录,证明又想已被占用

但是在nodejs中,大家都知道,各种的回调。简单的查询数据库都是异步的。你可能会这么写:

User.findOne({username: user.username}, function (err, doc) {
    if(!doc){ // 用户不存在,继续根据邮箱进行查询
        User.findOne({username: user.username}, function (err, doc) {
            if(doc){    // 邮箱存在返回错误
                req.flash('error', '邮箱已被占用!');
                res.redirect('/register');
            }else{
                //…… 进行注册操作
            }
        });
    }else{  // 用户存在,返回错误
        req.flash('error', '用户名已被占用')
        res.redirect('/register');
    }
});

已经嵌套了两层回调,如果是检查三个字段、四个字段、五个字段呢?后果不堪设想! 有没有优雅的实现方法呢?答案是有的!我们用到了async

优雅的实现方式

async是一个流程控制工具包,可以完美解决我们的问题。(关于async的详细介绍,请移步项目地址:https://github.com/caolan/async; 也可以参考这里:http://blog.csdn.net/jbboy/article/details/37667809 ,这里就不作悖述 )

很显然,根据我们的需求,两次的数据库查询判断并没有对顺序做要求,所以我们选择asyncparallel:并行执行多个函数,然后做统一的判断处理。

async.parallel() 方法接收两个参数,一个是执行的方法组,一个是回调。

async.parallel([
    function(callback){
        dosomething(function(){
            callback(null, result1);
        });
    },
    function(callback){
        dosomething(function(){
            callback(null, result2);
        });
    }
], function(err, results){
    console.log(results); // results 值为:{result1, result2}
})

其中第一个参数,可以是上面那种数组的形式,也可以是对象的形式:

{
    one: function(callback){
    },
    two: function(callback){
    }
}

实现方法:

async.parallel({
    username: function (callback) {
        User.findOne({username: user.username}, function (err, doc) {
            callback(null, doc);
        });
    },
    email: function (callback) {
        User.findOne({email: user.email}, function (err, doc) {
            callback(null, doc);
        });
    }
}, function (err, results) {
    if(results.username) {
        req.flash(config.constant.flash.error, '用户名已被占用');
        res.redirect('/join');
        return;
    }
    if(results.email){
        req.flash(config.constant.flash.error, '邮箱已被占用');
        res.redirect('/join');
        return;
    }
});

这样是不是简洁多了?

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

扫码关注云+社区

领取腾讯云代金券