- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在尝试使用 InnoDB 存储引擎对 MySQL 数据库表执行操作。此操作是一个 INSERT-or-UPDATE 类型的操作,其中我有一组传入的数据,并且表中可能已有一些数据必须更新。例如,我可能有这个表:
test_table
+-------+--------------+------+-----+---------+----------------+
| Field | Type | Null | Key | Default | Extra |
+-------+--------------+------+-----+---------+----------------+
| id | int(11) | NO | PRI | NULL | auto_increment |
| value | varchar(255) | NO | | NULL | |
+-------+--------------+------+-----+---------+----------------+
...以及一些示例数据:
+----+-------+
| id | value |
+----+-------+
| 1 | foo |
| 2 | bar |
| 3 | baz |
+----+-------+
现在,我想“合并”以下值:
2, qux
4, corge
我的代码最终发出以下查询:
BEGIN;
SELECT id, value FROM test WHERE id=2 FOR UPDATE;
UPDATE test SET id=2, value='qux' WHERE id=2;
INSERT INTO test (id, value) VALUES (4, 'corge');
COMMIT;
(我不太确定 SELECT ... FOR UPDATE
和 UPDATE
会发生什么,因为我正在使用 MySQL 的 Connector/J Java 库,并且只是在 updateRow
上调用 ResultSet
方法。对于为了便于讨论,我们假设上面的查询实际上是向服务器发出的查询。)
注意:上表是一个简单的例子来说明我的问题。真实的表更复杂,我在执行SELECT ... FOR UPDATE
时没有使用PK作为匹配的字段。因此,仅通过查看传入的数据并不清楚记录是否需要插入或更新。必须查阅数据库来确定是否使用 INSERT/UPDATE。
上述查询在大多数情况下都可以正常工作。然而,当有更多记录需要“合并”时,SELECT ... FOR UPDATE
和INSERT
行可以交错,我无法预测是否 SELECT ... FOR UPDATE
或INSERT
将发布以及按什么顺序发布。
结果是有时事务会死锁,因为一个线程锁定了UPDATE
表的一部分。操作并正在等待表锁(对于 INSERT
,它需要主键索引上的锁),而另一个线程已经获得了表锁主键(大概是因为它发出了 INSERT
查询),现在正在等待第一个线程持有的行锁(或更可能是页级锁)。
这是代码中唯一更新此表的位置,并且当前没有获取显式锁定。 UPDATE
的排序与 INSERT
似乎是问题的根源。
我可以想到几种可能性来“解决”这个问题。
LOCK TABLES
在合并过程之前获取表锁和 UNLOCK TABLES
之后。这显然不适用于 MariaDB Galera——我们 future 的产品可能会采用这种方式。INSERT
先查询。这将导致首先获取任何表级锁并避免死锁。#3 的问题在于,它在已经相当复杂的方法中需要更复杂的代码(“合并”操作本质上是复杂的)。更复杂的代码还意味着查询数量大约增加一倍( SELECT
来确定行 ID 是否已经存在,然后再使用另一个 SELECT ... FOR UPDATE
/UPDATE
来实际更新它)。该表存在一定程度的争用,因此我希望尽可能避免发出更多查询。
有没有办法强制MySQL获取表级锁而不使用LOCK TABLES
?也就是说,如果我们搬到加莱拉,这种方式会起作用吗?
最佳答案
我认为您可以通过获取一组行锁和间隙锁来完成您想要的操作:
START TRANSACTION;
SELECT id, value
FROM test
WHERE id in (2, 4) -- list all the IDs you need to UPSERT
FOR UPDATE;
UPDATE test SET value = 'qux' WHERE id = 2;
INSERT INTO test (id, value) VALUES (4, 'corge');
COMMIT;
SELECT
查询将锁定已存在的行,并为尚不存在的行创建间隙锁。间隙锁将阻止其他事务创建这些行。
关于mysql - 如何强制 MySQL 获取事务的表锁?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53308704/
我一直很难编辑我的 .htaccess 文件来一起做这三件事。我已经能够分别获得每个部分,但我只是不明白逻辑流程如何使它们全部工作。 这是我能够使用 bluehost support 上的演示进行整合
我制作的宏将模板工作簿保存为两个单独的文件。每个测试保存一个(位置 1、2、3 或 4),然后在另一个宏中使用每个测试的数据。第二个是保留用于备份的原始数据文件。现在的问题是每次我在每个位置运行测试并
我正在写一篇关于如何使用 OCaml 的模块系统而不是 Java 的 OO 系统(一个有趣的视角)的博客文章。我遇到了一些我不理解的关于强制的事情。下面是一个基本模块和两个包含它的模块: module
我有一段将被执行多次(5,000+)的代码,以及一个仅在第一次为真的 if 语句。我曾想过使用“FIRST”变量并每次都进行比较,但每次都检查它似乎是一种浪费,即使我知道它不需要。 bool FIRS
首先,我是 Perforce 的新手,我主要通过其文档进行学习。 因此,我们即将从 CVS 迁移到 Perforce,我最近学到了一个避免更改每个工作区的 P4CLIENT 的好方法,即在工作区根目录
我正在为一段代码编写测试,其中包含我试图涵盖的 IOException 捕获。 try/catch 看起来像这样: try { oos = new ObjectOutputStream(new
我正在尝试在新闻项目滚动之间添加延迟。我知道 $.each() 通过不等待动画完成来完成其工作,但我想知道如何制作它,以便一次向上滚动一个项目并等到最后一个动画完成后再继续在循环中。 $(functi
假设已经编写了一个方法,需要一个排序列表作为其输入之一。当然这将在代码中进行注释和记录,param 将被命名为“sortedList”,但如果有人忘记,则会出现错误。 有没有办法强制输入必须排序?我正
我正在尝试将传入请求重定向到 https://www.domain.com/和所有 https://www.domain.com/ {所有页面}并且没有什么麻烦。我试过的方法: 添加此行:Redire
我将如何实现以下内容: title_selection = raw_input("Please type in the number of your title and press Enter.\n%
我有一个登录表单,我需要强制关闭自动完成功能。我试过了 jquery: $('#login').attr("autocomplete", "off"); HTML: Javascript:docume
我想知道我应该怎么做才能强制从 dev 分支 merge 到我的 master 分支?使用“git merge dev”会导致很多冲突。但是,我不想单独处理它们。相反,我只是想使用我的 dev 分支中
当安装 Hl7.Fhir.DSTU2 和 Hl7.Fhir.R4 这两个 Nuget 包时,我们得到如下信息: DSTU2 包似乎在使用 Hl7.Fhir.Support.Poco 版本 3.4.0
我正在尝试让一个功能组件在 testFn 执行时强制重新渲染。我想使用状态来做到这一点(如果有更好的方法请说出来),这似乎成功地强制重新渲染但只有两次,然后什么都没有。 我构建了一个简单的演示来模拟这
默认情况下,g++ 似乎会省略未使用的类内定义方法的代码。示例 from my previous question : struct Foo { void bar() {} void baz(
我正在尝试使用 here 中介绍的技术使我的网站背景以比内容慢的速度滚动。我不希望背景固定,只希望更慢。 这是 HTML 的样子: .parallax { perspective: 1px;
我能找到的最相似的问题是 'how to create a row of scrollable text boxes or widgets in flutter inside a ListView?'
我有以下 eslint 配置: "object-curly-newline": ["error", { "ImportDeclaration": "never",
我正在使用 TinyMCE 插件并将 valid_elements 选项设置为: "a[href|target:_blank],strong/b,em/i,br,p,ul,ol,li" 即使没有列出数
您好,我想使用以下命令放置多行描述 p4 --field Description="MY CLN Header \\n my CLN complete description in two -thre
我是一名优秀的程序员,十分优秀!