[英]Mongoose findOneAndUpdate in a for loop using set if extist or put if doesn't
我在尝试在 for 循环中将 findOneAndUpdate 与 mongoose 一起使用时遇到并发问题。 不要关注给出的例子,而只关注它的方法。 假设我有这个架构:
var PlayerSchema = new Schema({
playername: String,
playerstats:[{
matchNumber: Number,
goals: {
type: Number,
default: 0
}
}]
})
假设我有一年的数组,我想用另一个用 for 循环迭代创建的数组来更新目标,我想做的是为 x = 0 到 5 设置一个 for 循环,其中 x 是 matchNumber 和 score[ x] 代表那场比赛的进球数(去年)。 如果 matchNumber 存在,我想查看 Player 文档,如果存在,我将使用 findOneAndUpdate 和 $set 来设置 score[x],但它不存在我想使用 $push 使用 score[x] 创建它. 让我们给出最容易理解的代码(“马拉多纳”将是我们的球员名字):
function iterateYears(){
years = ["2019", "2020"]
for (var k = 0; k < years.lenght; k++) {
updateScores(k)
}
}
function updateScores(k) {
var scores = []
for (var y = 0; y < 6; y++) {
scores.splice(y, 0, (y * 10) + k)
}
//at this point we would have the first time scores = [0, 10, 20, 30, 40] and second time scores = [1, 11, 21, 31, 41]
for (var x = 0; x < 6; x++) {
Player.findOneAndUpdate({playername: "Maradona", 'playerstats.matchNumber': x}, {$set: {'playerstats.$.goals': scores[x]}}, function(err, matchfound) {
if (err) {
console.log("Err found")
} else {
if (matchfound == null) {
Player.findOneAndUpdate({playername: "Maradona"}, {$push: {playerstats:{matchNumber: x, goals: scores[x]}}}).exec();
}
}
})
}
}
我最后会得到的是:
{
playername: "Maradona",
playerstats:[
{
matchNumber: 0,
goals: 1
},
{
matchnumber: 1,
goals: 11
},
{
matchNumber: 2,
goals: 21
},
{
matchNumber: 3,
goals: 31
},
{
matchNumber: 4,
goals: 41
}
}
但我得到的是这样的:
{
playername: "Maradona",
playerstats:[
{
matchNumber: 0,
goals: 1
},
{
matchNumber: 0,
goals: 0
},
{
matchnumber: 1,
goals: 11
},
{
matchnumber: 1,
goals: 10
},
{
matchNumber: 2,
goals: 21
},
{
matchNumber: 2,
goals: 20
},
{
matchNumber: 3,
goals: 31
},
{
matchNumber: 3,
goals: 30
},
{
matchNumber: 4,
goals: 41
},
{
matchNumber: 4,
goals: 40
}
}
或它们的混合。 发生这种情况是因为代码将尝试使用 $set 生成 findOneAndUpdate 并且当找不到 matchNumber (matchfound == null) 而不是使用 $push 执行 findOneAndUpdate 来创建它时,它将继续迭代,因此将尝试这样做与 k = 1 的值相同(第二年要澄清),并且需要再次移动到 findOneAndUpdate 与 $push 因为它不会被发现,所以分数 = (x * 10) + k 的值被创建之后,它将返回并使用 $push 和分数 = x * 10 进行 findOneAndUpdate。我知道这是它的正常工作,因为 mono 线程但我真的需要有所需的 output。 感谢所有人,很抱歉长时间阅读。
问题在于findOneAndUpdate
是异步的。 通过 for 循环的每次迭代都会调用Player.findOneAndUpdate
并向其传递一个回调 function。 findOneAndUpdate 调用立即返回,循环继续进行下一次迭代。
在未来的某个时刻,对 mongo 的查询完成并使用返回值调用回调,因为有多个异步调用暂存,它们完成的顺序没有严格定义。
您可以尝试在每次调用 findOneAndUpdate 时使用await
以使它们同步,这将序列化请求。
如果您使用的是 MongoDB 4.2,则可以使用更新的管道形式,以便能够在单个语句中插入和更新数组元素:
.update(
{playername: "Maradona"},
[{$set:{
playerstats:{
$concatArrays:[
{$filter:{
input:"$playerstats",
cond:{$ne:["$$this.matchNumber",x]}
}},
[{matchNumber: x, goals: scores[x] }]
]
}
}}]
)
为了确保这些以正确的顺序应用,您可以构建和排列所有要应用的操作,例如:
[
{updateOne:{
filter: {playername: "Maradona"},
update: [{$set:{
playerstats:{
$concatArrays:[
{$filter:{
input:"$playerstats",
cond:{$ne:["$$this.matchNumber",0]}
}},
[{matchNumber: 0, goals: 1 }]
]
}
}}]
}},
{updateOne:{
filter: {playername: "Maradona"},
update: [{$set:{
playerstats:{
$concatArrays:[
{$filter:{
input:"$playerstats",
cond:{$ne:["$$this.matchNumber",1]}
}},
[{matchNumber: 1, goals: 11 }]
]
}
}}]
}}
然后使用ordered=true
选项将该数组传递给bulkWrite 。
另请注意,如果您能够构建一个包含所有需要更新的分数的数组,您可以使用类似的语句在一次操作中更新单个玩家的所有分数,这应该会更好一些。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.