首页
学习
活动
专区
工具
TVP
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往

InnoDB 事务加锁分析

本文以 MySQL 数据库 InnoDB 引擎为例,为大家分析 InnoDB数据库引擎对默认的隔离级别可重复读(RR)的具体实现。...整文知识点介绍:事务4种隔离级别、不同隔离级别解决的问题、MVCC、锁的类型、加锁案例分析;阅读完整文相信大家对事务隔离级别的具体实现有了一定的认识。...Gap Locks(间隙锁):在索引记录之间加锁,或者在第一个索引记录之前加锁,或者在最后一个索引记录之后加锁。 Next-Key Locks:在索引记录上加锁,并且在索引记录之前的间隙加锁。...MySQL InnoDB 通过间隙锁解决了幻读问题。以下通过实际的案例分析来介绍InnoDB 是如果解决幻读问题的。 四、案例分析 在对SQL进行加锁分析前,需要明确表的结构和索引类型。...MySQL InnoDB的可重复读并不保证避免幻读,需要应用使用加锁读来保证。而这个加锁读使用到的机制就是next-key locks。

1.7K00

MySQL如何加锁避免并发事务导致的脏写?

多个事务同时并发更新一行数据时, 就有脏写问题。脏写绝对不允许,可依靠锁机制让多个事务更新一行数据的时候串行化,避免同时更新一行数据。 有个事务要来更新一行数据,他会先看这行数据有没有人加锁?...看到没人加锁,该事务就会创建一个锁,包含自己的trx_id和等待状态,然后把锁跟这行数据关联在一起。...因为事务A给那行数据加了锁,所以此时该数据被加锁。就不能再让别人访问了! 此时事务B也想更新那行数据,就检查当前这行数据是否被别人加锁,然后发现事务A抢先给这行数据加锁了,这可咋办?...事务B想,那我也加个锁,大不了等着排队呗,于是事务B也会生成一个锁数据结构,有其trx_id和等待状态,但因为在排队,所以等待状态就是true: 事务A这时更新完了数据,就会释放锁。...锁一旦释放,他就会去找,此时还有无别人对这行数据也加锁了呢?他发现事务B也加锁了。于是,就会把事务B的锁里的等待状态修改为false,然后唤醒事务B继续执行,此时事务B就获取到锁了:

1.3K10
您找到你想要的搜索结果了吗?
是的
没有找到

MySQL加锁范围分析

场景: 最近,遇到了一个关于mysql 加锁的问题,将当时的情形简化如下,有一个index_test表,表结构如下所示: mysql> CREATE TABLE `index_test` ( `priv_id...这里不得不提令一个概念:隔离级别 事务在并发执行的过程中会导致的几个问题如下: 脏读(Drity Read):当一个事务允许读取另外一个事务修改但未提交的数据时,就可能发生脏读(dirty reads...然后在网上搜索相关的资料,看看别人有没有遇到过这样的问题,在一篇关于MySQL加锁处理分析的blog中得到了启示,按照blog中组合七:id非唯一索引+RR的理论,gap锁的范围不仅跟被锁定的键有关,还跟主键有关...因此,在我们使用mysql加锁过程中,也首先需要搞清楚,我们的隔离级别是什么,是否开启了binlog等等,然后才能正确分析加锁的范围。...p=771 大神描述Mysql 加锁分析的blog http://hedengcheng.com/?

6.1K72

mysql语句加锁分析

其实并不能完全解决幻读问题, 这里可以参考另一篇博客[mysql事务隔离级别与加锁分析] SERIALIZABLE隔离级别下,需要分为两种情况讨论: 在系统变量autocommit=0时,也就是禁用自动提交时...为啥不加锁呢?因为启用自动提交意味着一个事务中只包含一条语句,一条语句也就没有啥不可重复读、幻读这样的问题了。...[](/images/mysql/ru_rc_table_scan.png) 2. `SELECT ... FOR UPDATE`进行加锁的情况与上边类似,只不过加的是+ XLock 3....语句一般情况下不加锁,不过当前事务在插入一条记录前需要先定位到该记录在B+树中的位置,如果该位置的下一条记录已经被加了gap锁(next-key锁也包含gap锁),那么当前事务会在该记录上加上一种类型为插入意向锁的锁...] [公众号:我们都是小青蛙 - MySQL加锁分析三部曲]

1.7K10

mysql语句加锁分析

其实并不能完全解决幻读问题, 这里可以参考另一篇博客[mysql事务隔离级别与加锁分析] SERIALIZABLE隔离级别下,需要分为两种情况讨论: 在系统变量autocommit=0时,也就是禁用自动提交时...为啥不加锁呢?因为启用自动提交意味着一个事务中只包含一条语句,一条语句也就没有啥不可重复读、幻读这样的问题了。...语句一般情况下不加锁,不过当前事务在插入一条记录前需要先定位到该记录在B+树中的位置,如果该位置的下一条记录已经被加了gap锁(next-key锁也包含gap锁),那么当前事务会在该记录上加上一种类型为插入意向锁的锁...SessionA 回滚, 此时 SessionB 和 SessionC 都试图继续执行插入操作, 都要加上 X 锁, 但两个Session都要等待对方的行锁, 所以出现了死锁 参考资料 [掘金小册-从根上理解MySQL...] [公众号:我们都是小青蛙 - MySQL加锁分析三部曲]

86930

MySQL 加锁处理分析

快照读,读取的是记录的可见版本 (有可能是历史版本),不用加锁。当前读,读取的是记录的最新版本,并且,当前读返回的记录,都会加上锁,保证其他事务不会再并发修改这条记录。...; 当前读:特殊的读操作,插入/更新/删除操作,属于当前读,需要加锁。 所有以上的语句,都属于当前读,读取记录的最新版本。并且,读取之后,还需要保证其他并发事务不能修改当前记录,对读取记录加锁。...相对而言,2PL比较容易理解,说的是锁操作分为两个阶段:加锁阶段与解锁阶段,并且保证加锁阶段与解锁阶段不相交。下面,仍旧以MySQL为例,来简单看看2PL在MySQL中的实现。 ?...深入理解MySQL如何加锁,有两个比较重要的作用: 可以根据MySQL加锁规则,写出不会发生死锁的SQL; 可以根据MySQL加锁规则,定位出线上产生死锁的原因; 下面,来看看两个死锁的例子 (一个是两个...结论:死锁的发生与否,并不在于事务中有多少条SQL语句,死锁的关键在于:两个(或以上)的Session加锁的顺序不一致。

3.5K61

MySQL加锁实战分析

ENGINE=InnoDB; insert into t values(0,0,0),(5,5,5), (10,10,10),(15,15,15),(20,20,20),(25,25,25); MySQL...的加锁规则是?...加锁的基本单位是next-key lock(前开后闭区间) 查找过程中访问到的对象才会加锁 索引上的等值查询,给唯一索引加锁的时候,next-key lock退化为行锁 索引上的等值查询,向右遍历时且最后一个值不满足等值条件的时候...: 根据规则1,加锁单位是next-key lock,Session A的加锁范围是(5, 10] 根据规则4,SessionA是等值查询,而id=10不满足等值条件,因此next-key lock退化为间隙锁...注意我们这里加的是读锁,如果你使用的是for update,MySQL会认为你要更新数据,因此会给主键索引上满足条件的行加上行锁。

1K30

MySQL语句加锁分析详解

当然,有时候因为MySQL具体的实现而导致一些情景下的加锁有些不太好理解,这就得我们死记硬背了~ 我们这里把语句分为3种大类:普通的SELECT语句、锁定读的语句、INSERT语句,我们分别看一下。...不过这里有一个小插曲: # 事务T1,REPEATABLE READ隔离级别下 mysql> BEGIN; Query OK, 0 rows affected (0.00 sec) mysql> SELECT...我们说语句一和语句二是MySQL中规定的两种锁定读的语法格式,而语句三和语句四由于在执行过程需要首先定位到被改动的记录并给记录加锁,也可以被认为是一种锁定读。...小贴士:上述步骤是在MySQL 5.7.21这个版本中验证的,不保证其他版本有无出入。...这个UPDATE语句是先对聚簇索引记录进行加锁,后对二级索引记录进行加锁,如果在不同事务中运行上述两个语句,可能发生一种贼奇妙的事情 —— 事务T2持有了聚簇索引记录的锁,事务T1持有了二级索引记录的锁

1.3K40

MySQL更新语句加锁

也就是说在一个事务中,不管有多少条增删改,都是在加锁阶段加锁,在 commit 后,进入解锁阶段,才会全部解锁。...组合三、id不唯一索引+RC 该组合中,id列不在唯一,而是个普通索引,那么当执行sql语句时,MySQL又是如何加锁呢?...这样做,保证了最后满足条件的记录加上锁,但是每条记录的加锁操作是不能省略的。 结论:若id列上没有索引,MySQL会走聚簇索引进行全表扫描过滤。由于是在MySQl Server层面进行的。...如何保证多次当前读返回一致的记录,那么就需要在多个当前读之间,其他事务不会插入新的满足条件的记录并提交。为了实现该结果,Gap锁就应运而生。...结论:在MySQL/InnoDB中,所谓的读不加锁,并不适用于所有的情况,而是和隔离级别有关。在Serializable隔离级别下,所有的操作都会加锁

2.1K20

MySQL 加锁和死锁解析

产生死锁的必要条件 多个并发事务(2个或者以上) 每个事物都持有了锁(或者是已经在等待锁) 每个事务都需要再继续持有锁(为了完成事务逻辑,还必须更新更多的行) 事物之间产生加锁的循环等待,形成死锁...总结 • 原则之一 要分析一个死锁,必须深入业务,了解整个事务的逻辑(闭门无法造车) • 原则之二` GAP锁很复杂,为了减少GAP锁,减少GAP导致的死锁,尽量选择Read Committed隔离级别...+ row based binlog,基本上能够解决所有问题,无需使用Repeatable Read) 适当的 减少Unique 索引,能够减少GAP锁导致的死锁(根据业务情况而定) • 原则之三 在MySQL...• 原则之四 RC隔离级别下,如果死锁中出现Next Key(Gap锁),说明表中一定存在unique索引 多语句事务产生的死锁,确保每条语句操作记录的顺序性,能够极大减少死锁 本文大多数都整理自《死锁...-何登成 - 管中窥豹——MySQL(InnoDB)死锁分析之道》

97320

mysql 事务

1)查看哪些引擎支持事务: SQL: Show engines; 2)查看表引擎类型: Show create table table_name; 3)查看是否自动提交: show variables...like ‘%autocommit%’; 4)事务开始的方法: a) mysql_autocommit(0); 如果程序在此处coredump,请检查是否connect db b) SQL:Set autocommit...=0; c) Begin work; d) Start transaction; 5)事务结束的方法: a) SQL:Commit/rollback b) Mysql_commit/mysqlrollback...c) 隐式事务,参考http://blog.csdn.net/blues1021/article/details/6329190 并发事务: 锁机制: 乐观锁:通过where条件控制、通过version...字段或自定义字段的值控制; update影响的行数:mysql_affected_rows的返回值,可根据它决定事务是否终止 悲观锁=排他锁 Select  * from table for update

2.4K10

MySQL 事务

MySQL中,事务具有以下四个特性,通常称为ACID特性: 1.原子性(Atomicity):事务中的所有操作要么全部执行成功,要么全部失败。...在MySQL中,可以使用以下命令来控制事务: •START TRANSACTION; or BEGIN;:开始一个新的事务。•COMMIT;:提交当前事务,使事务中的所有修改生效。...事务并发 事务并发是指多个事务同时执行,这可能会导致以下问题: 1.丢失更新:当两个或多个事务同时对相同的数据进行更新时,最后提交的事务可能会覆盖之前提交的事务所做的修改,导致之前的更新丢失。...MySQL支持以下四种事务隔离级别: 1.未提交读(Read Uncommitted):允许一个事务读取另一个事务未提交的数据。这可能导致脏读、不可重复读、幻读。...查看、设置MySQL事务隔离级别 -- 查看事务隔离级别 -- 使用系统变量查询 SELECT @@transaction_isolation; -- 5.7.20之前 SHOW VARIABLES

8910

Mysql事务

加锁读和非加锁读 按照是否加锁MySQL的读可以分为两种,一种是非加锁读,也称作快照读、一致性读,使用普通的select语句,这种情况下使用MVCC避免了脏读、不可重复读、幻读,保证了隔离性。...#共享锁读取 select...lock in share mode #排它锁读取 select...for update 加锁读在查询时会对查询的数据加锁(共享锁或排它锁)。...由于锁的特性,当某事务对数据进行加锁读后,其他事务无法对数据进行写操作,因此可以避免脏读和不可重复读。而避免幻读,则需要通过next-key lock。...因此,加锁读同样可以避免脏读、不可重复读和幻读,保证隔离性。...RR虽然避免了幻读问题,但是毕竟不是Serializable,不能保证完全的隔离 如果在事务中第一次读取采用非加锁读,第二次读取采用加锁读,则如果在两次读取之间数据发生了变化,两次读取到的结果不一样,因为加锁读时不会采用

1.7K10

MySQL 事务

**insert、update、delete**),现在 **MySQL** 中有 **InnoDB & NDB** 存储引擎支持事务。...MySQL InnoDB 中对隔离级别的支持 在 **MySQL** 中的 **InnoDB** 存储引擎支持的隔离级别与 **SQL92** 定义的基本一致,隔离级别越高,事务的并发度就越低。...如果满足的情况下,**MySQL** 会重新发起一次读操作,此时会读取行的最新版本并加锁。...这两个锁主要作用是在 **InnoDB** 中就可以支持更多粒度的锁;以及提高加锁的效率,如果没有意向锁的情况下准备给一张表加上表锁时必须先判断是否有其他事务锁定了其中的某行,有的情况下就不能加锁;也就是说需要去扫描整张表才能确定是否加表锁...在 **MySQL** 中有一个参数控制获取锁的等待时间,默认是 **50m**。

2.9K20

MySQL事务

几个并行执行的事务,其执行结果必须与按某一顺序串行执行的结果相一致 隔离性(Isolation):事务的执行不受其他事务的干扰,事务执行的中间结果对其他事务必须是透明的 持久性(Durability):...对于任意已提交事务,系统必须保证该事务对数据库的改变不被丢失,即使数据库出现故障 提交 默认隔离级别可重复读,若事务级别为读已提交,则终端二不commit的情况下可以读取到终端一的数据。...### 终端一 begin; 开启 insert into students (sname) value ("sss"); commit; 只有commit才表示执行成功 ### 终端二 mysql>...insert into students (sname) value ("kksk"); 测试语句 ### 终端二 mysql> select * from students; +----+-----...-+ | id | sname | +----+-------+ | 1 | Gage | | 2 | sss | | 4 | kksk | +----+-------+ 查看隔离级别 mysql

2.2K40

MySQL的锁机制和加锁原理

MySQL的锁机制和加锁原理 文章目录 MySQL的锁机制和加锁原理 1.行锁 2.表锁 3.页锁 4.乐观锁和悲观锁 4.1悲观锁 4.2乐观锁 5.1InnoDB锁的特性 6.Record Lock...Lock+Gap Lock 1.行锁 ​ 行级锁是Mysql中锁定粒度最细的一种锁,表示只针对当前操作的行进行加锁。...如果成功加锁,那么就可以对记录做修改,事务完成后就会解锁了。 其间如果有其他对该记录做修改或加排他锁的操作,都会等待我们解锁或直接抛出异常。...当表有多个索引的时候,不同的事务可以使用不同的索引锁定不同的行,另外,不论 是使用主键索引、唯一索引或普通索引,InnoDB 都会使用行锁来对数据加锁。...相对而言,2PL比较容易理解,说的是锁操作分为两个阶段:加锁阶段与解锁阶段,并且保证加锁阶段与解锁阶段不相交。下面,仍旧以MySQL为例,来简单看看2PL在MySQL中的实现。

94620
领券