- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
假设我们有一个简单的应用程序,用户可以在其中创建产品并对其进行评论。产品和评论的架构可以是:
var productSchema = new mongoose.Schema({
author_id: ObjectId,
description: String
});
var commentSchema = new mongoose.Schema({
product_id: ObjectId,
author_id: ObjectId,
message: String
});
我们希望确保每条评论都指向现有产品。这可以通过 mongoose pre save hook 轻松完成:
commentSchema.pre("save", function(next) {
Product.count({ _id: this.product_id }, function(err, count) {
if (err || !count) {
next(new Error("Could not find product"));
} else {
next();
}
});
});
此外,如果用户删除了产品,我们希望删除对该产品的所有评论。这可以使用 pre remove hook 轻松完成:
productSchema.pre("remove", function(next) {
Comment.remove({ product_id: this._id }, next);
});
但是如果用户 A 删除了一个产品,同时用户 B 对该产品发表了评论呢?
可能会发生以下情况:
Call pre save hook for new comment, and check if product exists
Call pre remove hook for product, and remove all comments
In pre save hook, done checking: product actually exists, call next
Comment saved
In pre remove hook, done removing comments: call next
Product removed
最终结果是我们有一个评论指向一个不存在的产品。
这只是导致这种情况发生的众多情况之一。如何防止这种极端情况?
最佳答案
似乎使用 Mongoose post hook
s 而不是 pre hook
s 可以解决问题:
commentSchema.post("save", function(comment) {
Product.count({ _id: comment.product_id }, function(err, count) {
if (err || !count) comment.remove();
});
});
productSchema.post("remove", function(product) {
Comment.remove({ product_id: product._id }).exec();
});
让我们通过考虑四种可能的情况(我能想到的)来看看为什么这可以解决问题:
1) Comment gets saved before product is removed
2) Comment gets saved after product is removed but before post remove hook
3) Comment gets saved after product is removed and while post remove hook is
executing
4) Comment gets saved after product is removed and post remove hook executed
------------------------------------------------------------------------
In case 1, after the product is removed, the comment will be removed in the post
remove hook.
In case 2, same, post remove hook will remove the comment.
In case 3, the comment post save hook will successfully remove the comment.
In case 4, same as case 3, post save hook removes the comment.
但是还有一个小问题:如果在产品被移除之后 post remove hook
执行之前发生了一些不好的事情怎么办?说停电或类似的事情。在这种情况下,我们最终会得到引用不存在产品的评论。为了解决这个问题,我们可以在产品上保留 pre remove hook
。这保证了仅在删除所有相关评论时才删除产品。然而,正如 OP 所指出的,这不能处理并发问题,这就是我们的 post remove hook
来救援的地方!所以我们两者都需要:
productSchema.pre("remove", function(next) {
var product = this;
Comment.remove({ product_id: product._id }, next);
});
productSchema.post("remove", function(product) {
Comment.remove({ product_id: product._id }).exec();
});
我希望是这样,但我仍然可以想到一个非常遥远的案例:如果在删除产品后保存评论并且 post remove hook
在评论之前执行 怎么办? >post save hook
执行(这将删除评论)灯熄灭!我们最终得到的评论是指不存在的产品!这种情况发生的几率非常低,但仍然......
如果有人能想到更好的并发处理方法,请改进我的答案或自己编写!
关于javascript - 使用 mongoose 中间件删除依赖文档时的并发问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42521550/
我有一个 .sln 文件,里面有几个项目。为了简单起见,让我们称它们为... 项目A 项目B 项目C ...其中 A 是引用 B 和 C 的主要项目。我的目标是更新我的构建脚本,为 ProjectA
我安装了 Magento,我想知道如何生成完整的 API 文档,例如 http://docs.magentocommerce.com/ 上的文档是使用 phpdoc 生成的。 Magento 中是否包
我通常使用jetbrains family ide。在为函数创建文档时非常有用,只需输入 /** 如何在创建文档时创建自定义标签,例如@date标签。 最佳答案 JavaScript、Java: st
我正在尝试使用 jOpenDocument library创建文档。我已经执行了创建电子表格的示例 - 代码编译并运行正常,但当我尝试使用 Excel Office 2012 或 Google Doc
如标题。 有没有介绍HTML DOM构造的图片? 最佳答案 DOM(文档 对象模型)从文档 节点开始。它被称为“根节点”。 观察下面的树(括号中对应的nodeType): [HTMLDocument]
我喜欢 ColdFusion Builder。但我不喜欢帮助只有 CF9 文档。有什么方法可以将其更改为拥有 ColdFusion 8 文档? 最佳答案 http://livedocs.adobe.c
这个问题在这里已经有了答案: What is the consequence of this bit of javascript? (4 个答案) 关闭 9 年前。 我看到一些 jQuery 脚本嵌
我有一个 XML 文件,其中包含需要在 Word 文档中填充的数据。 我需要找到一种方法来定义一个模板,该模板可用作从 XML 文件填充数据并创建输出文档的基线。 我相信有两种方法可以做到这一点。 创
我正在尝试查找有关如何使用 AVAudioEngine 的详细文档。有谁知道我在哪里可以找到它? 我找到了这个,但与文档丰富的 UI 内容相比,它似乎非常简陋。 https://developer.a
我对 Tensorflow 文档越来越感到恼火和沮丧。我在谷歌上搜索了有关 的文档 tf.reshape 我被定向到一个通用页面,例如 here 。我想查看 tf.reshape 的详细信息,而不是整
我正在学习本教程:http://moxleystratton.com/clojure/clojure-tutorial-for-the-non-lisp-programmer 然后遇到了这个片段: u
如何在 swagger 中为对象数组编写文档。这是我的代码,但我不知道如何访问对象数组中的数据。 { "first_name":"Sam", "last_name":"Smith",
是否有针对 Javascript 的 JavaDocs 之类的东西?当我在 netbeans IDE 中按 ctrl+space 时 写javascript,指定对象的javascript文档就出来了
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 5 年前。
我需要 JavaScript 中的 heredoc 之类的东西。你对此有什么想法吗?我需要跨浏览器功能。 我发现了这个: heredoc = '\ \ \ zzz\ \
WSDL 文档是包含一系列的,可描述某个 web service 的定义的,简单的 XML 文档 WSDL 文档结构 WSDL 文档用下表这些主要的元素来描述某个 web service 的
是否有 ocropus 的文档? 我正在寻找对以下功能的解释: make_SegmentPageByRAST(): segment() RegionExtractor(): setPageLines(
这个问题在这里已经有了答案: Understanding events and event handlers in C# (13 个回答) 4年前关闭。 我正在使用 NRECO 和 ffmpeg 对视
我正在尝试访问工作服务器以与名为 Spotfire 的应用程序一起使用。我的同事把这个传给我,现在已经休息了几个星期,我对他的建议有意见。 实际上,当我通过 localhost 运行我的 Web 应用
Elm 文档没有给出示例用法,因此很难理解类型规范的含义。在几个地方,我看到“a”用作参数标识符,例如 Platform.Cmd : map : (a -> msg) -> Cmd a -> Cmd
我是一名优秀的程序员,十分优秀!