找到内部回调的另一个查找(...),如何逃避回调地狱?

时间:2015-04-01 05:29:21

标签: javascript node.js callback mongoose mean

(首先:对不起,我不会说英语!)

我想在一个数组中返回3个发现的结果。 我的代码(下一个)运行良好,但我在回调地狱!

_Schema
  .static('retrieveAll', function(cb) {
    query = {};
    this.find(query, function(err, data) {
      if(err) {
        cb(err, null);
        return;
      }

      if(data)
        all = data;
      else
        all = [];

      _StoresModel.find(query).select('contact address').exec(function(err, data) {
        if(err) {
          cb(err, null);
          return;
        }

        if(data) {
          all = data.reduce(function(coll, item) {
            coll.push(item);
            return coll;
          }, all);
        }

        _CustomersModel.find(query).select('contact address').exec(function(err, data) {
          if(err) {
            cb(err, null);
            return;
          }

          if(data) {
            all = data.reduce(function(coll, item) {
              coll.push(item);
              return coll;
            }, all);
          }

          cb(null, all);          
        });
      });
    });
  });

我在FIND内的FIND内找到了一个FIND。 反正有没有改善这个?

解:

_Schema   .static(' retrieveAll',function(cb){     var model = this;

_async.parallel(
  { contacts: function(cb) {
      model.find({}).exec(cb);
    }
  , stores: function(cb) {
      _StoresModel.find({}).select('contact address').exec(cb);
    }
  , costumers: function(cb) {
      _CostumersModel.find({}).select('contact address').exec(cb);
    }
  }
, function(err, data) {
  if(err) {
    cb(err, null);
    return
  }

  var ret = [];
  if(data.contacts.length > 0) {
    ret = ret.concat(data.contacts);
  }
  if(data.stores.length > 0) {
    ret = ret.concat(data.stores);
  }
  if(data.costumers.length > 0) {
    ret = ret.concat(data.costumers);
  }

  cb(null, ret);
});

3 个答案:

答案 0 :(得分:1)

您可以尝试使用Promises

(未经测试)示例:

var RSVP = require('rsvp');
var all = [];

_Schema.static('retrieveAll', function(cb) {
    query = {};

    findPromise(this, query)
    .then(function (data) {
        all = data;
        return findPromise(_StoresModel, query, 'contact address');
    })
    .then(function (stores) {
        all = all.concat(stores);
        return findPromise(_CustomersModel, query, 'contact address');
    })
    .then(function (customers) {
        all = all.concat(customers);
        cb(null, all);
    })
    .catch(function (err) {
        cb(err, null);
    });
});

function findPromise(Model, query, select) {
    return new RSVP.Promise(function (resolve, reject) {
        Model.find(query).select(select || '*').exec(function (err, data) {
            return err ? reject(err) : resolve(data);
        });
    });
}

该示例使用RSVP,但还有其他承诺实施,例如Qbluebird

另请注意,您可以使用concat来连接数组,而不是使用reduce

答案 1 :(得分:0)

某些服务器端承诺库(如qbluebird)会大大清理您的代码并消除回调地狱的混乱局面。

答案 2 :(得分:0)

看看npm Async。它是一个很棒的库,可以在node.js上使用不同的模式。

如果存在按时间顺序排列的优先级或waterfall模式,如果它们都可以并行执行,您可能希望使用parallel