- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在编写一个 Web 应用程序,并且我一直在尝试将来自每个 Web 请求的 SQL 语句与带有 ISOLATION LEVEL REPEATABLE READ
的事务进行包装,以找出我的 Web 应用程序可能不执行的操作-可重复读取。我的计划是在不可重复读取的情况下不重试,而只是向用户报告服务器端错误 (500) 并记录信息(因为我预计这种情况非常罕见)。
与此同时,我在代码中的某些地方使用了显式锁定 (SELECT ... FOR UPDATE
) 以确保我正确地序列化访问并且不会导致不可重复的读取。
然而,将这两个想法结合在一起,给了我意想不到的结果。
下面是一个最小的例子:
+--------------------------------------------------+--------------------------------------------------+
| Session 1 | Session 2 |
+--------------------------------------------------+--------------------------------------------------+
| BEGIN; | |
| SET TRANSACTION ISOLATION LEVEL REPEATABLE READ; | |
| SELECT * FROM users WHERE id = 1 FOR UPDATE; | |
| (returns as expected) | |
+--------------------------------------------------+--------------------------------------------------+
| | BEGIN; |
| | SET TRANSACTION ISOLATION LEVEL REPEATABLE READ; |
| | SELECT * FROM users WHERE id = 1 FOR UPDATE; |
| | (blocks as expected) |
+--------------------------------------------------+--------------------------------------------------+
| UPDATE users SET name = 'foobar' WHERE id = 1; | |
| COMMIT; | |
| (works as expected) | |
+--------------------------------------------------+--------------------------------------------------+
| | ERROR: could not serialize access due |
| | to concurrent update |
+--------------------------------------------------+--------------------------------------------------+
我的期望是,由于 session 2 在该 SELECT
语句之前没有进行任何读取,并且由于该语句仅在 session 1 完成更新后返回,因此 session 2 应该会看到表的更新版本,这将使它成为可重复读取的。
我认为,Postgres 很可能在运行 BEGIN
时采用一个版本,而不是在它为第一个 SELECT
获取锁时采用一个版本。
我的问题:
最佳答案
来自 "13.2.2. Repeatable Read Isolation Level" :
UPDATE
,DELETE
,SELECT FOR UPDATE
, andSELECT FOR SHARE
commands behave the same asSELECT
in terms of searching for target rows: they will only find target rows that were committed as of the transaction start time. However, such a target row might have already been updated (or deleted or locked) by another concurrent transaction by the time it is found. In this case, the repeatable read transaction will wait for the first updating transaction to commit or roll back (if it is still in progress). If the first updater rolls back, then its effects are negated and the repeatable read transaction can proceed with updating the originally found row. But if the first updater commits (and actually updated or deleted the row, not just locked it) then the repeatable read transaction will be rolled back with the messageERROR: could not serialize access due to concurrent update
because a repeatable read transaction cannot modify or lock rows changed by other transactions after the repeatable read transaction began.
是的,如果 BEGIN
是指交易的开始,那么您的理解似乎是正确的。不,这不是错误,而是按预期和记录工作。
据我所知,默认情况下,READ COMMITTED
应该可以执行您想要的操作。请注意,在客户端 1 中提交第一个事务后,SELECT FOR UPDATE
将被阻止,直到客户端 2 提交或回滚,因为它的 SELECT FOR UPDATE
现在已成功。因此客户端 2 中的第一个事务将读取相同的值(除非它自己更改它们)直到事务结束。
Client 1 | Client 2
------------------------------------------------+------------------------------------------------
BEGIN TRANSACTION; |
SET TRANSACTION ISOLATION LEVEL READ COMMITTED; |
SELECT * FROM users WHERE id = 1 FOR UPDATE; |
| BEGIN TRANSACTION;
| SET TRANSACTION ISOLATION LEVEL READ COMMITTED;
| SELECT * FROM users WHERE id = 1 FOR UPDATE;
| -- blocks
UPDATE users SET name = 'foobar' WHERE id = 1; |
COMMIT; |
| -- name = 'foobar' is read
BEGIN TRANSACTION; |
SET TRANSACTION ISOLATION LEVEL READ COMMITTED; |
SELECT * FROM users WHERE id = 1 FOR UPDATE; |
-- blocks |
| SELECT * FROM users WHERE id = 1 FOR UPDATE;
| -- name = 'foobar' is read
| COMMIT;
UPDATE users SET name = 'foobaz' WHERE id = 1; |
-- name = 'foobaz' is written |
关于postgresql - 隔离级别和显式锁定 : Unexpected Serialization Error,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54173405/
我刚刚更新了 Ruby,现在我在尝试启动 compass 时遇到以下错误: Encoding::CompatibilityError on line ["28"] of /usr/local/Cell
关闭。这个问题不符合Stack Overflow guidelines .它目前不接受答案。 我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。 关闭 6 年前。
关闭。这个问题需要debugging details .它目前不接受答案。 编辑问题以包含 desired behavior, a specific problem or error, and th
按照目前的情况,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引发辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the
我正在尝试在我的 iOS 应用程序中开发可折叠/ Accordion 式的功能。这将是您可以在网站上找到的典型 FAQ 类型功能。我想点击标题,然后显示详细信息。 因为这是帮助部分,只有几个项目,我认
我正在尝试设计一个基于 REST 的 Web 服务来与我正在开发的农场动物管理系统进行交互。 为了详细说明问题,我收藏了动物 属于一个农场。每只动物都有自己的信息——例如姓名、身份证号、品种年龄等。因
我有 3 种不同的表单,其中复选框数量不同,每个部分基本上代表一个表单,因此当用户选择该部分中的复选框时,它会显示他们在该部分的总金额中 checkout 了多少 HTML
我有一份 32 页的 PDF 版家谱。与其将家谱全部放在一个非常大的 PDF 页面上(这是我想要的),不如将其格式化为一组 8 个单独的美国信纸大小的页面应该在整个宽度上缝合; 4 行这样就完成了树。
指SASS implementation for Java? : 在 Maven 目标编译包中自动编译 compass-style.org 样式表的最佳方法是什么? 我不想发送太多的自编译库,也不想通
鉴于以下 XAML... 我正在寻找一种绑定(bind) ComboBox、Button 和 Command 的方法,以便当 ComboBox 的值更改时,在 Command 上调用 CanExe
在玩具应用程序中,我有一个显示所有帖子标题的“帖子”模板。当您单击每个标题时,我不想直接进入“显示” View ,而是直接内联展开该帖子的其余内容。 我考虑过让 postRoute 重用 postsR
我需要一些使用 Twitter Bootstrap 或其他响应式框架的自定义 Swagger-UI 实现。需要在我的移动设备上使用这样的 UI 测试我的 API,但 swagger-ui 不能很好地扩
我正在做一个项目,我真的在尝试编写面向对象的 JavaScript 代码。我刚刚开始阅读Douglas Crockford's JavaScript: The Good Parts我很快开始意识到用
在 C# 中,我通过执行以下操作来加密文本数据(请注意我正在以 block ( block )的形式加密数据): public string EncryptData(string pu
我正在构建一个社交网站,该网站将向全世界公开 REST API (WCF WebAPI),以便任何开发人员都能够为该网站创建客户端应用程序、将其与其他服务集成等。 我想为 API 实现 Faceboo
我是一名优秀的程序员,十分优秀!