InnoDB行锁和表锁的分析

2024-03-18 14:48
文章标签 分析 innodb 行锁 表锁

本文主要是介绍InnoDB行锁和表锁的分析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!



InnoDB行锁和表锁的分析

 
  1. InnoDB行锁是通过给索引上的索引项加锁来实现的,这一点MySQL与Oracle不同,后者是通过在数据块中对相应数据行加锁来实现的。 
  2. InnoDB这种行锁实现特点意味着:只有通过索引条件检索数据,InnoDB才使用行级锁,否则,InnoDB将使用表锁! 
  3. 在实际应用中,要特别注意InnoDB行锁的这一特性,不然的话,可能导致大量的锁冲突,从而影响并发性能。
  4. 下面通过一些实际例子来加以说明。 
  5. (1)在不通过索引条件查询的时候,InnoDB使用的是表锁,而不是行锁 

在如表20-9所示的例子中,开始tab_no_index表没有索引:
表20-9         InnoDB存储引擎的表在不使用索引时使用表锁例子
 
  1. session_1     
  2. mysql> set autocommit=0;  
  3. Query OK, 0 rows affected (0.00 sec)  
  4. mysql> select * from tab_no_index where id = 1 ;  
  5. +------+------+  
  6. | id   | name |  
  7. +------+------+  
  8. | 1    | 1    |  
  9. +------+------+  
  10. 1 row in set (0.00 sec)  
  11.   
  12. session_2  
  13. mysql> set autocommit=0;  
  14. Query OK, 0 rows affected (0.00 sec)  
  15. mysql> select * from tab_no_index where id = 2 ;  
  16. +------+------+  
  17. | id   | name |  
  18. +------+------+  
  19. | 2    | 2    |  
  20. +------+------+  
  21. 1 row in set (0.00 sec)  
  22.  
  23. session_1:   
  24. mysql> select * from tab_no_index where id = 1 for update;   
  25. +------+------+   
  26. | id   | name |   
  27. +------+------+   
  28. | 1    | 1    |   
  29. +------+------+   
  30. 1   
  31. session_2:  
  32. mysql> select * from tab_no_index where id=2 for update;  
  33. ERROR 1205 (HY000): Lock wait timeout exceeded; try restarting transaction  
在如表20-9所示的例子中,看起来session_1只给一行加了排他锁,但session_2在请求其他行的排他锁时,却出现了锁等待!原因就是在没有索引的情况下,InnoDB只能使用表锁。当我们给其增加一个索引后,InnoDB就只锁定了符合条件的行,如表20-10所示。
创建tab_with_index表,id字段有普通索引:
 
  1. mysql> create table tab_with_index(id int,name varchar(10)) engine=innodb
  2. Query OK, 0 rows affected (0.15 sec) 
  3. mysql> alter table tab_with_index add index id(id); 
  4. Query OK, 4 rows affected (0.24 sec) 
  5. Records: 4  Duplicates: 0  Warnings: 0 
表20-10    InnoDB存储引擎的表在使用索引时使用行锁例子

 

     
  1. session_1    
  2. mysql> set autocommit=0
  3. Query OK, 0 rows affected (0.00 sec) 
  4. mysql> select * from tab_with_index where id = 1 ; 
  5. +------+------+ 
  6. | id   | name | 
  7. +------+------+ 
  8. | 1    | 1    | 
  9. +------+------+ 
  10. 1 row in set (0.00 sec) 
  11. session_2 
  12. mysql> set autocommit=0
  13. Query OK, 0 rows affected (0.00 sec) 
  14. mysql> select * from tab_with_index where id = 2 ; 
  15. +------+------+ 
  16. | id   | name | 
  17. +------+------+ 
  18. | 2    | 2    | 
  19. +------+------+ 
  20. 1 row in set (0.00 sec) 
  21. session_1 
  22. mysql> select * from tab_with_index where id = 1 for update; 
  23. +------+------+ 
  24. | id   | name | 
  25. +------+------+ 
  26. | 1    | 1    | 
  27. +------+------+ 
  28. 1 row in set (0.00 sec) 
  29.  
  30. session_2    
  31. mysql> select * from tab_with_index where id = 2 for update; 
  32. +------+------+ 
  33. | id   | name | 
  34. +------+------+ 
  35. | 2    | 2    | 
  36. +------+------+ 
  37. 1 row in set (0.00 sec) 
 
(2)由于MySQL的行锁是针对索引加的锁,不是针对记录加的锁,所以虽然是访问不同行的记录,但是如果是使用相同的索引键,是会出现锁冲突的。应用设计的时候要注意这一点。
在如表20-11所示的例子中,表tab_with_index的id字段有索引,name字段没有索引:

 

 
  1. mysql> alter table tab_with_index drop index name; 
  2. Query OK, 4 rows affected (0.22 sec) 
  3. Records: 4  Duplicates: 0  Warnings: 0 
  4. mysql> insert into tab_with_index  values(1,'4'); 
  5. Query OK, 1 row affected (0.00 sec) 
  6. mysql> select * from tab_with_index where id = 1
  7. +------+------+ 
  8. | id   | name | 
  9. +------+------+ 
  10. | 1    | 1    | 
  11. | 1    | 4    | 
  12. +------+------+ 
  13. 2 rows in set (0.00 sec) 
表20-11    InnoDB存储引擎使用相同索引键的阻塞例子
 
  1. session_1    
  2. mysql> set autocommit=0
  3. Query OK, 0 rows affected (0.00 sec) 
  4. session_2 
  5. mysql> set autocommit=0
  6. Query OK, 0 rows affected (0.00 sec) 
  7. session_1    
  8. mysql> select * from tab_with_index where id = 1 and name = '1' for update; 
  9. +------+------+ 
  10. | id   | name | 
  11. +------+------+ 
  12. | 1    | 1    | 
  13. +------+------+ 
  14. 1 row in set (0.00 sec) 
  15. 虽然session_2访问的是和session_1不同的记录,但是因为使用了相同的索引,所以需要等待锁: 
  16. mysql> select * from tab_with_index where id = 1 and name = '4' for update; 
  17. 等待 
(3)当表有多个索引的时候,不同的事务可以使用不同的索引锁定不同的行,另外,不论是使用主键索引、唯一索引或普通索引,InnoDB都会使用行锁来对数据加锁。
在如表20-12所示的例子中,表tab_with_index的id字段有主键索引,name字段有普通索引:

 

 
  1. mysql> alter table tab_with_index add index name(name); 
  2. Query OK, 5 rows affected (0.23 sec) 
  3. Records: 5  Duplicates: 0  Warnings: 0 
表20-12    InnoDB存储引擎的表使用不同索引的阻塞例子

 

     
  1.  session_1  ·           
  2. mysql> set autocommit=0
  3. Query OK, 0 rows affected (0.00 sec) 
  4. session_2 
  5. mysql> set autocommit=0
  6. Query OK, 0 rows affected (0.00 sec) 
  7.  session_1 
  8. mysql> select * from tab_with_index where id = 1 for update; 
  9. +------+------+ 
  10. | id   | name | 
  11. +------+------+ 
  12. | 1    | 1    | 
  13. | 1    | 4    | 
  14. +------+------+ 
  15. 2 rows in set (0.00 sec) 
  16.   
  17.      
  18. Session_2使用name的索引访问记录,因为记录没有被索引,所以可以获得锁: 
  19. mysql> select * from tab_with_index where name = '2' for update; 
  20. +------+------+ 
  21. | id   | name | 
  22. +------+------+ 
  23. | 2    | 2    | 
  24. +------+------+ 
  25. 1 row in set (0.00 sec) 
  26.      
  27. 由于访问的记录已经被session_1锁定,所以等待获得锁。: 
  28. mysql> select * from tab_with_index where name = '4' for update; 
 
(4)即便在条件中使用了索引字段,但是否使用索引来检索数据是由MySQL通过判断不同执行计划的代价来决定的,如果MySQL认为全表扫描效率更高,比如对一些很小的表,它就不会使用索引,这种情况下InnoDB将使用表锁,而不是行锁。因此,在分析锁冲突时,别忘了检查SQL的执行计划,以确认是否真正使用了索引。关于MySQL在什么情况下不使用索引的详细讨论,参见本章“索引问题”一节的介绍。
在下面的例子中,检索值的数据类型与索引字段不同,虽然MySQL能够进行数据类型转换,但却不会使用索引,从而导致InnoDB使用表锁。通过用explain检查两条SQL的执行计划,我们可以清楚地看到了这一点。
例子中tab_with_index表的name字段有索引,但是name字段是varchar类型的,如果where条件中不是和varchar类型进行比较,则会对name进行类型转换,而执行的全表扫描。
当我们用范围条件而不是相等条件检索数据,并请求共享或排他锁时,InnoDB会给符合条件的已有数据记录的索引项加锁;对于键值在条件范围内但并不存在的记录,叫做“间隙(GAP)”,InnoDB也会对这个“间隙”加锁,这种锁机制就是所谓的间隙锁(Next-Key 锁)。
举例来说,假如emp表中只有101条记录,其empid的值分别是 1,2,...,100,101,下面的SQL:

 

 
  1. Select * from  emp where empid > 100 for update; 
是一个范围条件的检索,InnoDB不仅会对符合条件的empid值为101的记录加锁,也会对empid大于101(这些记录并不存在)的“间隙”加锁。
InnoDB使用间隙锁的目的,一方面是为了防止幻读,以满足相关隔离级别的要求,对于上面的例子,要是不使用间隙锁,如果其他事务插入了empid大于100的任何记录,那么本事务如果再次执行上述语句,就会发生幻读;另外一方面,是为了满足其恢复和复制的需要。有关其恢复和复制对锁机制的影响,以及不同隔离级别下InnoDB使用间隙锁的情况,在后续的章节中会做进一步介绍。
很显然,在使用范围条件检索并锁定记录时,InnoDB这种加锁机制会阻塞符合条件范围内键值的并发插入,这往往会造成严重的锁等待。因此,在实际应用开发中,尤其是并发插入比较多的应用,我们要尽量优化业务逻辑,尽量使用相等条件来访问更新数据,避免使用范围条件。
还要特别说明的是,InnoDB除了通过范围条件加锁时使用间隙锁外,如果使用相等条件请求给一个不存在的记录加锁,InnoDB也会使用间隙锁!
在如表20-13所示的例子中,假如emp表中只有101条记录,其empid的值分别是1,2,......,100,101。
表20-13    InnoDB存储引擎的间隙锁阻塞例子
session_1 session_2
mysql> select @@tx_isolation;
+-----------------+
| @@tx_isolation  |
+-----------------+
| REPEATABLE-READ |
+-----------------+
1 row in set (0.00 sec)
mysql> set autocommit = 0;
Query OK, 0 rows affected (0.00 sec)
mysql> select @@tx_isolation;
+-----------------+
| @@tx_isolation  |
+-----------------+
| REPEATABLE-READ |
+-----------------+
1 row in set (0.00 sec)
mysql> set autocommit = 0;
Query OK, 0 rows affected (0.00 sec)
当前session对不存在的记录加for update的锁:
mysql> select * from emp where empid = 102 for update;
Empty set (0.00 sec)
 
 
这时,如果其他session插入empid为201的记录(注意:这条记录并不存在),也会出现锁等待:
mysql>insert into emp(empid,...) values(201,...);
阻塞等待
Session_1 执行rollback:
mysql> rollback;
Query OK, 0 rows affected (13.04 sec)
 
 
由于其他session_1回退后释放了Next-Key锁,当前session可以获得锁并成功插入记录:
mysql>insert into emp(empid,...) values(201,...);
Query OK, 1 row affected (13.35 sec)

注:关于Innodb什么情况下使用行锁,什么情况使用表锁,上面的例子介绍的通俗易通,适合初级DBA学习。原来出自http://brilon.iteye.com/blog/433726,欢迎交流沟通,谢谢!

这篇关于InnoDB行锁和表锁的分析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/822679

相关文章

Springboot中分析SQL性能的两种方式详解

《Springboot中分析SQL性能的两种方式详解》文章介绍了SQL性能分析的两种方式:MyBatis-Plus性能分析插件和p6spy框架,MyBatis-Plus插件配置简单,适用于开发和测试环... 目录SQL性能分析的两种方式:功能介绍实现方式:实现步骤:SQL性能分析的两种方式:功能介绍记录

最长公共子序列问题的深度分析与Java实现方式

《最长公共子序列问题的深度分析与Java实现方式》本文详细介绍了最长公共子序列(LCS)问题,包括其概念、暴力解法、动态规划解法,并提供了Java代码实现,暴力解法虽然简单,但在大数据处理中效率较低,... 目录最长公共子序列问题概述问题理解与示例分析暴力解法思路与示例代码动态规划解法DP 表的构建与意义动

C#使用DeepSeek API实现自然语言处理,文本分类和情感分析

《C#使用DeepSeekAPI实现自然语言处理,文本分类和情感分析》在C#中使用DeepSeekAPI可以实现多种功能,例如自然语言处理、文本分类、情感分析等,本文主要为大家介绍了具体实现步骤,... 目录准备工作文本生成文本分类问答系统代码生成翻译功能文本摘要文本校对图像描述生成总结在C#中使用Deep

Redis主从/哨兵机制原理分析

《Redis主从/哨兵机制原理分析》本文介绍了Redis的主从复制和哨兵机制,主从复制实现了数据的热备份和负载均衡,而哨兵机制可以监控Redis集群,实现自动故障转移,哨兵机制通过监控、下线、选举和故... 目录一、主从复制1.1 什么是主从复制1.2 主从复制的作用1.3 主从复制原理1.3.1 全量复制

Redis主从复制的原理分析

《Redis主从复制的原理分析》Redis主从复制通过将数据镜像到多个从节点,实现高可用性和扩展性,主从复制包括初次全量同步和增量同步两个阶段,为优化复制性能,可以采用AOF持久化、调整复制超时时间、... 目录Redis主从复制的原理主从复制概述配置主从复制数据同步过程复制一致性与延迟故障转移机制监控与维

Redis连接失败:客户端IP不在白名单中的问题分析与解决方案

《Redis连接失败:客户端IP不在白名单中的问题分析与解决方案》在现代分布式系统中,Redis作为一种高性能的内存数据库,被广泛应用于缓存、消息队列、会话存储等场景,然而,在实际使用过程中,我们可能... 目录一、问题背景二、错误分析1. 错误信息解读2. 根本原因三、解决方案1. 将客户端IP添加到Re

Redis主从复制实现原理分析

《Redis主从复制实现原理分析》Redis主从复制通过Sync和CommandPropagate阶段实现数据同步,2.8版本后引入Psync指令,根据复制偏移量进行全量或部分同步,优化了数据传输效率... 目录Redis主DodMIK从复制实现原理实现原理Psync: 2.8版本后总结Redis主从复制实

锐捷和腾达哪个好? 两个品牌路由器对比分析

《锐捷和腾达哪个好?两个品牌路由器对比分析》在选择路由器时,Tenda和锐捷都是备受关注的品牌,各自有独特的产品特点和市场定位,选择哪个品牌的路由器更合适,实际上取决于你的具体需求和使用场景,我们从... 在选购路由器时,锐捷和腾达都是市场上备受关注的品牌,但它们的定位和特点却有所不同。锐捷更偏向企业级和专

Spring中Bean有关NullPointerException异常的原因分析

《Spring中Bean有关NullPointerException异常的原因分析》在Spring中使用@Autowired注解注入的bean不能在静态上下文中访问,否则会导致NullPointerE... 目录Spring中Bean有关NullPointerException异常的原因问题描述解决方案总结

python中的与时间相关的模块应用场景分析

《python中的与时间相关的模块应用场景分析》本文介绍了Python中与时间相关的几个重要模块:`time`、`datetime`、`calendar`、`timeit`、`pytz`和`dateu... 目录1. time 模块2. datetime 模块3. calendar 模块4. timeit