Sequelize Promises - Chaining Promises和Bluebird Promise.each,Promise.map,Promise.all等

时间:2016-09-19 22:36:42

标签: sql node.js promise sequelize.js bluebird

我的用户控制器中有一个索引操作,其中我尝试连续执行两项操作,而不是执行所需的res.json()方法,直到它们都有机会完成。

我有一个加入用户的友情加入模型。一列是frienderId,一列是friendedId。在下面的索引函数中,期望的结果是我最终会得到一个SINGLE用户对象,这是一个朋友。下面有承诺的原因是因为用户可能在frienderId列或friendedId列中,所以我必须基本上解析req.user的朋友的两种可能情况。

问题是归还的最后一件事,I.E。 res.json(result),总是返回正确的朋友,但是,它始终在所需对象之前或之后返回一个空数组[]。我知道这个空数组来自两个没有找到结果的承诺场景之一。是否有某种类型的方法而不是像sayyy..reject那样的解决方法?如果被召唤,整个承诺都不会被推到承诺阵列?然后我可以测试if(!friended){reject}或if(!friender){reject}

我理解。每个承诺都会列出每个承诺并给出每个承诺的结果。但是,我不希望每个承诺的结果,我只想访问每个承诺运行的副作用。

谢谢!

index: function(req, res) {
    var promiseArray = [];

    promiseArray.push(new Promise(function(resolve) {
      user.findById(req.user.id).then(function(user) {
        user.getFrienders({
          where: {
            username: req.body.username
          }
        }).then(function(friender) {
          resolve(friender[0])
        })
      })
    }));

    promiseArray.push(new Promise(function(resolve) {
      user.findById(req.user.id).then(function(user) {
        user.getFriendeds({
          where: {
            username: req.body.username
          }
        }).then(function(friended) {
          resolve(friended[0])
        })
      })
    }));

    Sequelize.Promise.map(promiseArray, function(result) {
      res.json(result);
    });
}

1 个答案:

答案 0 :(得分:2)

首先,您对Promise的使用有点多余。由于findById()函数返回一个promise,所以你需要做的就是处理它。您无需创建new Promise。你可以说promiseArray.push(user.findById ...)。你几乎正在使用then。传递给then的函数是在解析promise时调用的函数,在这种情况下,在findById函数完成并返回一个值之后。 then本身返回一个使用其回调返回的值解析的promise。因此,无论你希望在完成所有操作后最终在promiseArray中获得什么值,我都假设friended[0]friender[0]您需要在上一个then函数中返回该值。在第一个then函数中,您还应该说return user.findFriends( ...

接下来需要做的是处理从findById函数返回的承诺被拒绝的情况。你可以用两种方式做到这一点。首先,您可以将第二个回调传递给then,在拒绝承诺的情况下将调用该回调,并将拒绝的原因作为其参数。您可以将thenthen一起关注,而不是将两个函数传递到catch,这是我个人喜欢做的,并将您的拒绝回调传递给catchcatch还返回一个使用其回调返回的值解析的promise。您可以选择返回传入的原因,或者您想要的任何值。因此,如果错误被拒绝,那么catch返回的内容将最终出现在promises数组中。

对于承诺,阅读the documentation可能对您有所帮助。不可否认,我知道在我刚开始学习时我的承诺有些困难,但在完成文档并编写实例之后,你就会掌握它,看看有多么有用的承诺。

如果是我,这就是我重构代码的方式

    var frienderPromise = user.findById(req.user.id)
          .then(function(user) {
            return user.getFrienders({
              where: { username: req.body.username }
            });
          })
          .then(function (friender) {
            return friender[0];
          })
          .catch(function (reason) {
            return reason; // or whatever else you want to end up in the promisesArray on rejection
          });

    var friendedPromise = user.findById(req.user.id)
          .then(function(user) {
            return user.getFriendeds({
              where: { username: req.body.username }
            });
          })
          .then(function (friended) {
            return friended[0];
          })
          .catch(function (reason) {
            return reason;
          });

    promiseArray.push(frienderPromise);
    promiseArray.push(friendedPromise);

你可以保持其他一切。最后,您仍然可以像当前一样致电Sequalize。虽然它应该是return Sequalize...。如果您不想保留不良结果,也可以考虑使用Promise.filter代替.map。所以你可以做类似

的事情
    return Sequelize.Promise.filter(promiseArray, function(result) {
      if (/* make sure it's result you want */)
        res.json(result);
    });

就个人而言,我会更进一步,将所有承诺业务移除到另一个与res无关的功能,并用

完成
    return Sequelize.Promise.filter(promiseArray, function(result) {
      return result === /* result you want */;
    });

然后,一旦进入index功能,就可以调用

    return friendsPromiseFunction().then(function (results) {
      res.json(results);
    })

它会使事情变得更加清洁。希望这可以帮助。如果您有疑问,请告诉我。