- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
MySQL InnoDB 在事务中的非唯一索引上使用下一个键锁定,从而锁定扫描索引前后的间隙(顺便说一下,MySQL 手册未能以清晰的方式传达,手册页next-key locks 表示只有扫描索引之前的间隙被锁定:http://dev.mysql.com/doc/refman/5.7/en/innodb-record-level-locks.html。
但是,我无法理解这背后的全部原理......
使用的设置:
CREATE TABLE test (a int, b int, index (a));
INSERT INTO test VALUES (5,5), (10,10), (15,15);
第一个连接的客户端启动事务 A 并发出以下 UPDATE
查询:
UPDATE test set b = 10 where a = 10;
从下一个传入连接开始事务 B 运行以下查询会得到以下结果:
INSERT INTO test VALUES(5,5); //On hold
INSERT INTO test VALUES(9,9); //On hold
INSERT INTO test VALUES(14,14); //On hold
INSERT INTO test VALUES(4,4); //Works
INSERT INTO test VALUES 15,15); //Works
UPDATE test SET a = 1 WHERE a = 5; //Works
UPDATE test SET a = 8 WHERE a = 5; //On hold
UPDATE test SET a = 7 WHERE a = 15; //On hold
UPDATE test SET a = 100 WHERE a = 15; //Works
事务 B 似乎无法在 a 为 [5,15) (5 incl. - 15 excl.) 的位置插入行,也无法修改现有行并将 a 设置为 (5, 15) (5 excl. - 15 excl.) .
现在,将 a
列更改为具有 PRIMARY KEY
:
ALTER TABLE test DROP INDEX a;
ALTER TABLE test ADD PRIMARY KEY (a);
在事务 B 中重做上面的运行现在给出以下结果(插入第 5 行和第 15 行给出了关于重复键的错误,这就是它们不包括在内的原因):
INSERT INTO test VALUES(9,9); //Works
INSERT INTO test VALUES(14,14); //Works
INSERT INTO test VALUES(4,4); //Works
INSERT INTO test VALUES(10,10); //On hold
UPDATE test SET a = 1 WHERE a = 5; //Works
UPDATE test SET a = 8 WHERE a = 5; //Works
UPDATE test SET a = 7 WHERE a = 15; //Works
UPDATE test SET a = 100 WHERE a = 15; //Works
UPDATE test SET a = 10 WHERE a = 15; //On hold
UPDATE test SET a = 100 WHERE a = 10; //On hold
使用主键的行为似乎完全可以理解,我不质疑它(即使缺少间隙锁,使用间隙锁来防止幻读的基本原理,也不会阻止幻读)。我根本不质疑这种行为,我只是很难理解处理常规索引的方式以及为什么以不同的方式处理它们。
问题:
SELECT
查询不应返回的规则在隔离级别 REPEATABLE READ
中整个事务中的不同结果)或者是因为 InnoDB 推断用户可能希望插入接近查询的结果(这将然后成为用户的启发式服务)?第三个原因可能是整个系统原则似乎是锁定查询结果的任何行,而 InnoDB 这样做是没有考虑的(然后会遵守一些关于并发规则的总体原则) .来自 http://dev.mysql.com/doc/refman/5.7/en/innodb-next-key-locking.html似乎仅当 WHERE
子句具有像 a > 10
这样有意义的条件时,才使用 next-key 锁定来防止幻读,但如果是这样,为什么当 WHERE
子句专门针对某些行时,是否也应用了下一键锁定?也许有几个不相干的原因?a = 5
被锁定用于 INSERT
而不是用于 UPDATE
?就好像同时有两种锁定原则在起作用,一种是锁定对现有行的修改,另一种是锁定插入,现有行 a = 5
未锁定,但行的插入a = 5
被锁定。这是正确的吗?如果是这样,为什么索引 5 包含在用于插入的间隙锁中?我的 MySQL 版本是 5.5.24
,我使用默认隔离级别 REPEATABLE READ
。
最佳答案
你的问题太多了。 :)
我不是数据库专家,我只是给你一些提示。
一个)。索引约束不一定是唯一性的。当条件列没有索引或不是唯一索引时,MySQL 使用间隙锁。因为主键是唯一性索引,所以它只是锁定选中的记录。
b).当您更新索引列时,实际上记录需要重新索引。因为innodb使用了聚簇索引,也就是说记录在主索引B+树的叶子上。所以当需要找到一个位置放置更新后的 inode 时,数据库需要授予锁请求。
UPDATE test SET a = 10 WHERE a = 15; //On hold
a = 15 处没有锁,但是当您想将索引放置在 a = 10 处时,那里有一个现有锁。所以它成立。
关于mysql - MySQL InnoDB 下一键锁定中唯一索引和非唯一索引之间差异背后的基本原理,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26215071/
我对 Java 并发性比较陌生(还没有阅读 JCIP,但它在我的列表中!)并且我有一个关于锁定行为的问题。具体来说,Java 是锁定对象的引用,还是锁定对象本身? 代码示例(不是 sscce,因为我不
我的团队使用 TortoiseSVN 编写版本控制代码。有时,有人使用“获取锁定”选项。是否有可能看到解决方案中的锁? 最佳答案 http://tortoisesvn.net/docs/nightly
我在使用 SVN 时遇到了一个小问题。 当我跑 svn stat我明白了: ~ some/dir 当我跑 svn commit -m "test"我明白了:svn: working copy
我启用了 jenkins 安全性,认为它会提示我创建一个帐户。我尝试在 c:/program files/jenkins 中删除和编辑我的 config.xml 文件,但我不确定如何在没有访问权限的情
实现与 S3 结合使用的简单锁定机制的推荐方法是什么? 我想做的例子: 通过对象 ID 获取锁 从 S3 读取对象 修改数据 将对象写入 S3 释放锁 理想情况下寻找基于云的锁定机制。我可以在本地使用
找到这个here : 一般来说,在以下任何情况下,请考虑在列上创建索引: 索引列上存在引用完整性约束,或者列。索引是避免全表锁的一种方法,否则,如果您更新父表主键,则需要,合并到父表中,或从父表中删除
在我的程序中,我将把每个“ block ”数据存储在一个单独的文件中。多个线程都会读取和写入各种文件,我想避免因未正确同步而可能出现的问题。本质上,我想要一个设置,其中每个文件的行为就好像它有自己的
我想使用此script作为资源,通过使用Windows API(重置管理器)与Go for Windows中的内容相同 到目前为止,我的代码是 Rstrtmgr := syscall.NewLazyD
这里的问题是:“这些选择中的哪一个对于线程安全选择的剧院具有最佳性能?” public static List lockList = initializeLocks(); public boolean
我有一个侧面菜单,单击图标时打开,单击页面或单击菜单上的项目时关闭。我正在尝试实现锁定,因此当单击锁定图标时,即使您单击菜单项或页面,菜单也不会关闭。 我能够将图标从锁定图标更改为解锁图标,但我在停止
使用 TRueType 字体编写 SDL 程序。我调用 TTF_Init() 来初始化 TTF 并使用 TTF_OpenFont( name, size ) 打开我的字体。 我有一个例程,可以使用以下
我正在尝试调试基于运行 FreeRTOS 的 STM32F3 uC 的应用程序。我已在应用程序的线程上下文中的随机位置手动将 PSP 设置为无效值(例如 0),希望触发 memManageFault/
我有以下 C# 代码: 1. List bandEdgeList; 2. 3. bandEdgeList = CicApplication.BandEdgeCache.Where(r
我正在用骰子制作游戏。这个想法是持有/锁定骰子。我把骰子做成按钮,这样现在就可以点击它们了。示例:我抛出一个“6”和一个“1”。我点击“6”,所以现在只会抛出“1”。 我对这个有点迷失了,我需要创建
我正在使用以下代码下载约 200mb 的播客并将其写入文档目录: var podcastRequest = NSURLRequest(URL: audioUrl) NSURLConnection.se
下面的类 DoStuff 启动一个线程并同步以保护监听器对象在 null 时不被访问。 现在,当从外部访问 DoStuff 类函数 setOnProgressListener() 时,我遇到了问题,因
我正在编写一个使用巨大背景 Canvas 的网站。我试图锁定浏览器调整大小处理程序以避免滚动问题(背景越界等) 这是我第一次做一个完整的后台网站。任何有关优化的建议(png 大小 580.72 KB
我是 C# 和线程的新手,我有这个问题要解决: 我有一个处理一些数据的线程,它会不时(必要时)触发我在启动线程之前设置的事件方法 (DataProcessor)。该线程位于专有 dll 中。所以我不能
我正在使用相机,我使用的是文档中给出的完全相同的示例: http://developer.android.com/resources/samples/ApiDemos/src/com/example/
我有几个座位可供用户预订。同一时间,只有一个用户可以参与预订过程,这样同一个座位就不会被多个用户预订。在我的 Java 代码中,我使用了“synchronized”关键字来完成它。这行得通。 但是,现
我是一名优秀的程序员,十分优秀!