数据库行锁、表锁、记录锁、间隙锁、临键锁
行锁
行锁就是一次锁一行或者多行记录,mysql的行锁是基于索引加载的,所以行锁是要加在索引响应的行上,即命中索引。
// 假设表格table的b列为索引字段,则以下更新语句只会锁b='b'的行
update table set a = 'a' where b = 'b'
行锁的特征:锁冲突概率低,并发性高,但是会有死锁的情况出现。
表锁
表锁就是一次锁一整张表的记录,在表被锁定期间,其他事务不能对该表进行操作,必须等当前表的锁被释放后才能进行操作。表锁响应的是非索引字段,即全表扫描。
// 假设表格table的c列为非索引字段,则以下更新语句会锁table整张表
update table set a = 'a' where c = 'c'
表锁的特征:由于表锁每次都是锁一整张表,所以表锁的锁冲突几率特别高,表锁不会出现死锁的情况。
记录锁
记录锁锁的是表中的某一条记录,记录锁的出现条件必须是精准命中索引并且索引是唯一索引,如主键id
// 以下语句,id为唯一索引
update table set a = 'a' where id = 1
间隙锁
间隙锁又称之为区间锁,每次锁定都是锁定一个区间,隶属行锁。既然间隙锁隶属行锁,那么,间隙锁的触发条件必然是命中索引的,当我们查询数据用范围查询而不是相等条件查询时,查询条件命中索引,并且没有查询到符合条件的记录,此时就会将查询条件中的范围数据进行锁定(即使是范围库中不存在的数据也会被锁定)
间隙锁只会出现在可重复读的事务隔离级别中,mysql5.7默认就是可重复读
// id=1和id=3的行都是在数据库中存在的
update table set a = 'a' where id > 1 and id < 3
上述代码中,所锁定的区间就是 (1,3]这个区间,不包含1,但是包含3,并且不包含4,也就是说这里是一个左开右闭的区间。
临键锁
mysql的行锁默认就是使用的临键锁,临键锁是由记录锁和间隙锁共同实现的,上面我们学习间隙锁时,间隙锁的触发条件是命中索引,范围查询没有匹配到相关记录。而临键锁恰好相反,临键锁的触发条件也是查询条件命中索引,不过,临键锁有匹配到数据库记录
例子:
假如数据库中只有三条数据1、5、7,当修改范围为1~8时,则锁定的区间为(1,+∞),锁定的不单是查询范围,并且还锁定了当前范围的下一个范围区间,此时,查询的区间8,在数据库中是一个不存在的记录值,并且,如果此时的查询条件是小于或等于8,也是一样的锁定8到后面的区间。
为什么临键锁后匹配会这样呢?mysql的索引是基于B+树实现的,每个树节点上都有多个元素,即关键字数,当我们的索引树上只有1、5、7时,我们查询1~8,这个时候由于树节点关键字中并没有8,所以就把8到正无穷的区间范围都给锁定了。
那么,如果我们数据库中id有1、5、7、10,此时我们再模糊匹配id为1~8的时候,由于关键字中并没有8,所以找比8大的,也就找到了10,根据左开右闭原则,此时10也是被锁定的,但是id为11的记录还是可以正常进行插入的。我们的锁都是基于索引的,而mysql中索引的底层是使用的B+树,我们了解了B+树的特性后,就更容易理解很多遇到锁的问题了。
参考:https://zhuanlan.zhihu.com/p/52678870