Mongoose 解决 Query was already executed 问题

本文已整理到 Github,地址 👉 blog

如果我的内容帮助到了您,欢迎点个 Star 🎉🎉🎉 鼓励鼓励 :) ~~

我希望我的内容可以帮助你。现在我专注于前端领域,但我也将分享我在有限的时间内看到和感受到的东西。


当给定查询执行两次时,Mongoose 会抛出 "Query was already executed"(查询已执行)错误。对此最常见的解释是您正在混合 await 和回调。

await Model.updateMany({}, { $inc: { count: 1 } }, function (err) {})
// "MongooseError: Query was already executed"

这是因为 Mongoose 在收到回调或 await 时执行查询。如果您使用 await 并传递回调,则此查询将执行两次。

或者:

Model.updateMany({}, { $inc: { count: 1 } }, function (err) {}).then(() => {})

此查询执行两次。一次是因为回调,一次是因为 then() 方法。

解决方案是跳过传递回调。在 Mongoose 中不需要回调,因为 Mongoose 支持 promisesasync/await

await Model.updateMany({}, { $inc: { count: 1 } })
// or
Model.updateMany({}, { $inc: { count: 1 } }).then(() => {})

但如果我们想执行两次查询呢?可以使用 clone() 方法:

let query = Model.findOne()

await query

// 抛出 "MongooseError: Query was already executed" 错误
await query

// ✅
await query.clone()
  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
当使用Mongoose修改一个已经存在的文档时,如果修改后的字段值与其他文档的唯一键值(如email)冲突了,就会报错`MongoError: E11000 duplicate key error`。这是因为MongoDB要求唯一键值必须唯一,不能重复。 为了解决这个问题,可以考虑使用`findOneAndUpdate`方法来修改文档,而不是直接使用`updateOne`或`updateMany`方法。`findOneAndUpdate`方法会先查询符合条件的文档,如果找到了就更新它,否则就创建一个新文档。这样就可以避免唯一键值冲突的问题。 以下是使用`findOneAndUpdate`方法修改文档的示例代码: ```javascript async function updateUser(id, updates) { try { const user = await User.findOneAndUpdate({ _id: id }, updates, { new: true, // 返回更新后的文档 runValidators: true, // 执行验证器 context: 'query' // 将验证器应用于更新操作 }); return user; } catch (err) { if (err.name === 'MongoError' && err.code === 11000) { // 唯一键值冲突错误处理 throw new Error('Email already exists'); } else { throw err; } } } ``` 在上面的代码中,我们使用`findOneAndUpdate`方法来更新文档。如果更新后的文档与其他文档的唯一键值冲突了,就会抛出`MongoError`异常,我们可以在catch块中捕获这个异常并进行处理。在这里,我们抛出一个自定义的错误信息来提示用户。 需要注意的是,在使用`findOneAndUpdate`方法时,我们需要将`runValidators`和`context`选项设置为true,这样才能执行文档验证器并将其应用于更新操作。这可以确保更新后的文档符合Schema定义的规则。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值