- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我有以下场景:
我正在使用一个主 servlet,它:
- 从数据库池中获取连接
- 将自动提交设置为 false
- 执行通过应用程序层的命令:如果全部成功,在“finally”语句中将自动提交设置为真,并关闭连接。否则如果发生异常,rollback()。
在我的数据库 (mysql/innoDb) 中,我有一个“历史”表,其中包含行列:
id(主键) |用户名|日期 |话题 |锁定
“锁定”列的默认值为“假”,它用作标记特定行是否锁定的标志。
每一行都特定于一个用户(正如你可以从用户名列中看到的那样)
所以回到场景:
-->Ulc1 发送命令以更新他从数据库中获取日期“D”和主题“T”的历史记录。
-->Ulc2 发送 same 命令以更新 来自数据库的相同 日期“D”和相同的历史记录 同一时间的主题“T”。
我想实现一个 mysql/innoDB 锁定系统,使任何到达的线程都能执行以下检查:
此行的“锁定”列是否为真?
这两种 mysql 锁定技术中的哪一种实际上允许第二个到达的线程读取锁定列的“更新”值来决定采取 wt 操作?
我应该使用 “更新” 或 “锁定共享模式”?
这个场景解释了我想要完成的事情:
- Ulc1 线程先到达:列“锁定”为假,将其设置为真并继续更新过程
- Ulc2 线程到达时 Ulc1 的事务仍在处理中,即使该行通过 innoDb 功能锁定,它也不必等待但实际上读取列锁定的"new"值是“真”,因此实际上不必等到 Ulc1 事务提交来读取“锁定”列的值(无论如何到那时该列的值已经被重置为 false)。
我对这两种锁定机制不是很有经验,到目前为止我的理解是 LOCK IN SHARE MODE 允许其他事务读取锁定的行,而 FOR UPDATE 甚至不允许读取。但是这个读取是否获得了更新的值?或者第二个到达的线程必须等待第一个线程提交然后读取值?
欢迎任何有关在这种情况下使用哪种锁定机制的建议。
此外,如果有更好的方法来“检查”行是否已被锁定(除了使用真/假列标志),请告诉我。
谢谢
解决方案
(Jdbc 伪代码示例基于@Darhazer 的 回答)
表:[ id(主键)|用户名|日期 |话题 |锁定]
connection.setautocommit(false);
//transaction-1
PreparedStatement ps1 = "Select locked from tableName for update where id="key" and locked=false);
ps1.executeQuery();
//transaction 2
PreparedStatement ps2 = "Update tableName set locked=true where id="key";
ps2.executeUpdate();
connection.setautocommit(true);// here we allow other transactions threads to see the new value
connection.setautocommit(false);
//transaction 3
PreparedStatement ps3 = "Update tableName set aField="Sthg" where id="key" And date="D" and topic="T";
ps3.executeUpdate();
// reset locked to false
PreparedStatement ps4 = "Update tableName set locked=false where id="key";
ps4.executeUpdate();
//commit
connection.setautocommit(true);
最佳答案
LOCK IN SHARE MODE 将允许第二个线程读取该值,但实际值将是查询(已提交读)或事务(可重复读取)开始之前的值(因为 MySQL 使用多版本控制; 并且第二个事务必须看到的内容由隔离级别定义)。因此,如果第一个事务在读取时未提交,则将读取旧值。
在您的场景中,最好有 1 个事务使用 select for update 锁定记录,另一个事务处理记录,在提交/回滚时,第三个事务解锁记录。
带有 select for update 的第二个线程事务将等待第一个线程完成,然后读取实际值并决定不继续其他事务,而是通知用户记录已锁定。
为避免死锁,请确保您正在使用唯一索引执行select for update
。
示例代码:
connection.setautocommit(false);
//transaction-1
PreparedStatement ps1 = "Select locked from tableName for update where id="key" and locked=false);
ps1.executeQuery();
//transaction 2
PreparedStatement ps2 = "Update tableName set locked=true where id="key";
ps2.executeUpdate();
connection.setautocommit(true); // here we allow other transactions / threads to see the new value
connection.setautocommit(false);
//transaction 3
PreparedStatement ps3 = "Update tableName set aField="Sthg" where id="key" And date="D" and topic="T";
ps3.executeUpdate();
// probably more queries
// reset locked to false
PreparedStatement ps4 = "Update tableName set locked=false where id="key";
ps4.executeUpdate();
//commit
connection.setautocommit(true);
关于mysql - "FOR UPDATE"v/s "LOCK IN SHARE MODE": Allow concurrent threads to read updated "state" value of locked row,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/9037345/
我有一个阅读器,其中包含有关 51*51 网格的信息,其中网格上的每个点都由 f32 表示。 .我想将这些数据读入一个向量,以便我可以轻松处理它: pub fn from_reader(reader:
我重新启动了 SQL Server 2005 并运行了统计 IO 的查询。 我得到了这些结果:表“xxx”。扫描计数 1,逻辑读取 789,物理读取 3,预读读取 794,... 预读读取数是读取并放
在 CLHS 中,我为 :read-only x 读到:“当 x 为真时,这指定不能更改此插槽;它将始终包含构造时提供的值。” 我可以做到这一点(CCL、SBCL): CL-USER> (defstr
让我们考虑一下这句话(Total Store Ordering): reads are ordered before reads, writes before writes, and reads be
我正在开发一个 SMTP 库,它使用缓冲读取器通过网络读取行。 我想要一种安全的方式来从网络读取数据,而不依赖于 Rust 内部机制来确保代码按预期工作。具体来说,我想知道 Read trait 是否
我不清楚所有这些读取字符串函数之间的关系。嗯,很明显clojure.core/read-string可以读取 pr[n] 输出的任何序列化字符串甚至 print-dup .也很清楚clojure.ed
所以我做了这个功能,就像倒计时一样。我想在倒计时减少时读取命令。我的大问题是让 read() 在倒计时减少时等待输入。如您所见,我尝试使用 select() 但在第一个 printf 之后("time
这是我vue3+echart5 遇到的报错:Cannot read properties of undefined (reading ‘type‘) 这个问题需要搞清楚两个关键方法: toRaw: 作
下图中,左边是C代码,右边是未优化的LLVM IR形式。 The Figure 在 IR 上运行 MemoryDependenceAnalysis 可查找内存依赖性。原始代码及其 IR 等效代码中
这个问题在这里已经有了答案: Read values into a shell variable from a pipe (17 个答案) 关闭 3 年前。 我一直在尝试像这样从程序输出中读取环境变
当我输入相同的整数时,如何将整数转换为与使用 read(0,buff,nbytes) 获得的缓冲区相同的值/编码字符?我正在尝试编写类似 read() 的东西,但用整数数据代替读取到缓冲区的文件描述符
This question already has answers here: Closed 2 years ago. Read input in bash inside a while loop (
我正在尝试处理来自 MySQL 数据库的一些数据(主要是 double 值)。我收到此错误消息: Invalid attempt to access a field before calling Re
我正在制作一个简单的 TCP/IP 套接字应用 这样做有什么不同: DataInputStream in = new DataInputStream(clientSocket.getInputStre
我操作API服务器。 手机APP访问API服务器时,有时会出现该异常。 我尝试在测试服务器上进行测试,但无法重现。(我改变了apache和tomcat的连接时间。) 有什么问题?? 我该如何解决这个问
我在段落末尾使用“阅读更多”只是为了提醒像P.T.O一样的用户 为什么会有问题? 最佳答案 您必须明白,许多屏幕阅读器用户不会等到整个页面都读给他们听。他们使用键盘快捷键在页面中导航。 JAWS(可以
我已将我的 Angular 应用程序从 12 版本升级到 13 版本。我在单元测试运行期间开始遇到此错误。 Chrome Headless 94.0.4606.61 (Windows 10) AppC
我正在尝试为以下组件编写一个。我正在使用 queryParams 然后使用 switchmap 来调用服务。这是 url 的样子: http://localhost:4200/test-fee/det
我的代码有什么问题? Uncaught TypeError: Cannot read properties of undefined (reading 'remove') 和 Uncaught Typ
我在我的 React 应用程序中遇到了这个问题。 类型错误:无法读取未定义的属性(读取“requestContent”) 我在我的应用程序中使用 commercejs。代码指向 isEmpty=!ca
我是一名优秀的程序员,十分优秀!