我的用户控制器中有一个索引操作,其中我尝试连续执行两项操作,而不是执行所需的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);
});
}
答案 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
,在拒绝承诺的情况下将调用该回调,并将拒绝的原因作为其参数。您可以将then
与then
一起关注,而不是将两个函数传递到catch
,这是我个人喜欢做的,并将您的拒绝回调传递给catch
。 catch
还返回一个使用其回调返回的值解析的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);
})
它会使事情变得更加清洁。希望这可以帮助。如果您有疑问,请告诉我。