前言
mysql按锁的范围分三种
- 表级锁:开销小,加锁快;不会出现死锁,锁定粒度大,发生锁冲突概率最高,并发度最低。
- 行级锁:开销大,加锁慢,会出现死锁,锁定粒度小,发生锁冲突的概率最低,并发度最高。
- 页面锁:开销和加锁时间界于表锁和行锁之间,会出现死锁,锁定粒度界于表锁和行锁之间,并发度一般。
从上述三种锁的特点来看,很难说哪种锁更好,只能就具体应用的特点来说哪种锁更合适。比如,MyISAM和MEMORY引擎采用的是表级锁;InnoDB引擎既支持行级锁,也支持表级锁,但默认情况下采用行级锁。
InnoDB的加锁模式【mysql的锁】InnoDB实现了以下两种类型的行锁。
- 共享锁(S):允许一个事务读一行,阻止其他事务获得相同数据的排他锁,也叫读锁。
- 排他锁(X):允许获得排他锁的事务更新数据,阻止其他事务取得相同数据集的共享锁与排他锁,也叫写锁。
意向锁的主要目的是为了使得 行锁 和 表锁 共存,事务在申请行锁前,必须先申请表的意向锁,成功后再申请行锁。
意向锁分为意向共享锁和意向排他锁。
- 意向共享锁(IS):事务打算给数据行加行共享锁,事务在给一个数据行加共享锁之前必须先去的该表的意向共享锁
- 意向排他锁(IX):事务打算给数据行加行排他锁,事务在给一个数据行加排他锁前必须先取得该表的意向排他锁。
右侧代表请求锁模式,下侧代表当前锁模式 | X | IX | S | IS |
---|---|---|---|---|
X | 冲突 | 冲突 | 冲突 | 冲突 |
IX | 冲突 | 兼容 | 冲突 | 兼容 |
S | 冲突 | 冲突 | 兼容 | 兼容 |
IS | 冲突 | 兼容 | 兼容 | 兼容 |
意向锁是表级锁,但是却表示事务正在读或写某一行记录,而不是整个表,所以意向锁之间不会产生冲突,真正的冲突在加行锁时检查。
加锁方法 意向锁是InnoDB自动加的,不需要用户干预。
隐式上锁
- 对于UPDATE,DELETE和INSERT语句,InnoDB会自动给设计数据集加排他锁;
- 对于普通SELETE语句,INNODB不会加任何锁;
- InnoDB会根据隔离级别在需要的时候自动加锁;
显式上锁
select * from tableName lock in share mode; //读锁 select * from tableName for update; //写锁
解锁
- 提交事务(commit)
- 回滚事务(rollback)
- kill阻塞进程
上读锁实例
事务A 事务B begin; select * from teacher where id = 2 lock in share mode; // 上读锁 select * from teacher where id = 2; // 可以正常读取 update teacher set name = 3 where id =2; // 可以更新操作 update teacher set name = 5 where id =2; // 被阻塞 commit; update teacher set name = 5 where id =2; // 更新操作成功
事务A 事务B begin; select * from teacher where id = 2 for update; // 上写锁 select * from teacher where id = 2; // 可以正常读取 update teacher set name = 3 where id =2; // 可以更新操作 update teacher set name = 5 where id =2; // 被阻塞 rollback; update teacher set name = 5 where id =2; // 更新操作成功
因为InnoDB有MVCC机制(多版本并发控制),可以使用快照读,而不会被阻塞。
InnoDB行锁实现方式行锁(Record Lock)
行锁总是会去锁住索引记录,如果InnoDB存储引擎表建立的时候没有设置任何一个索引,这时InnoDB存储引擎会使用隐式的聚簇索引来进行锁定。
间隙锁(Gap Lock)
当我们用范围条件而不是相等条件检索数据,并请求共享或排他锁时,InnoDB会给符合条件的已有数据记录的索引项加锁;对于键值在条件范围内但并不存在的记录,叫做“间隙(GAP)”,InnoDB也会对这个“间隙”加锁,这种锁机制就是所谓的间隙锁(Next-Key锁)。
优点:解决了事务并发的幻读问题
不足:因为query执行过程中通过范围查找的话,他会锁定争个范围内所有的索引键值,即使这个键值并不存在。
间隙锁有一个致命的弱点,就是当锁定一个范围键值之后,即使某些不存在的键值也会被无辜的锁定,而造成锁定的时候无法插入锁定键值范围内任何数据。在某些场景下这可能会对性能造成很大的危害。
Next-key Lock 锁
同时锁住数据+间隙锁
在Repeatable Read隔离级别下,Next-key Lock 是默认的行记录锁定算法。
假如teacher表中只有101条记录,其id值分别是1-101,SQL语句如下
Select * from teacher where id〉 100 for update;
这是一个范围条件检索,InnoDB不仅会对符合条件的id值为101的记录加锁,也会对id大于101(不存在的记录)的“间隙”加锁。
乐观锁与悲观锁 - 乐观锁(Optimistic Lock):假设不会发生并发冲突,只在提交操作时检查是否违反数据完整性。 乐观锁不能解决脏读的问题。
乐观锁, 顾名思义,就是很乐观,每次去拿数据的时候都认为别人不会修改,所以不会上锁,但是在更新的时候会判断一下在此期间别人有没有去更新这个数据,可以使用版本号等机制。乐观锁适用于多读的应用类型,这样可以提高吞吐量,像数据库如果提供类似于write_condition机制的其实都是提供的乐观锁。 - 悲观锁(Pessimistic Lock):假定会发生并发冲突,屏蔽一切可能违反数据完整性的操作。
悲观锁,就是很悲观,每次去拿数据的时候都认为别人会修改,所以每次在拿数据的时候都会上锁,这样别人想拿这个数据就会被阻塞直到它拿到锁。传统的关系型数据库里边就用到了很多这种锁机制,比如行锁,表锁等,都是在做操作之前先上锁。
总结
因此,在不同的隔离级别下,InnoDB 处理 SQL 时需要的锁是不同的。
推荐阅读
- mysql|mysql 插入语句语法_SQL 插入数据(INSERT INTO 语句)
- 数据库|TiDB Online DDL 在 TiCDC 中的应用丨TiDB 工具分享
- 进阶PHP月薪30k|PHP+Mysql如何实现数据库增删改查
- MySQL|数据库进阶_完整性、多表增删改查和数据库优化(MySQL)
- MySQL|MySQL数据库-对数据表的基本操作
- MySQL的四种事务隔离级别
- BUG小王子|JDBC及Mybatis常见BUG
- spring|springboot简单小项目练习之增删改查-exercisePro01
- 笔记|DBUtil工具