MySQL高级 - 锁
1 概述
锁是计算机协调多个进程或线程并发访问某一资源的机制。
2 全局锁
2.1 概念
全局锁就是对整个数据库实例加锁,加锁后整个实例就处于只读状态,后续的DML、DDL语句,已经更新操作的事物提交语句都将被阻塞。
例如: 全库的数据备份
2.2 使用
(1)给数据库的表加锁
flush tables with read lock;
(2)备份数据库
mysqldump - u{账号} -p{密码} {数据库名}> {文件名}.sql
(3)全局锁解锁
unlock tables
2.3 特点
(1) 主表备份,备份期间更新操作停止。
(2) 从表备份,备份期间从表不能执行主库同步过来的二进制日志,导致主从延迟。
在innoDB引擎中,我们可以 通过 --single-transaction 参数来完成不加锁的一致性数据备份
mysqldump --single-transaction -u{账号} -p{密码} {数据库名} > {文件名}.sql
3 表级锁
3.1 概念
每次操作锁住整张表。锁定粒度大,发生锁冲突的概率最高,并发度最低。应用子啊MyISAM、InnoDB、BDB等存储引擎中。
3.2 分类
(1)表锁
(2)元数据锁(metadata lock, MDL)
(3)意向锁
3.3 表锁
3.3.1 表共享读锁(read lock)
(1)加锁
lock tables {表名} read;
(2)解锁
unlock tables;
(3)理解
客户端1 对 表1 加 读锁
客户端1 对 表1 只读 不能写
其他客户端 对 表1 只读 不能写
3.3.2 表共享写锁(write lock)
(1)加锁
lock tables {表名} write;
(2)解锁
unlock tables;
(3)理解
客户端1 对 表1 加 写锁
客户端1 对 表1 可读写
其他客户端 对 表1 不可读写
3.4 元数据锁(meta data lock, MDL)
3.4.1 概念
MDL加锁过程是系统自动控制,在访问一张表时自动加上。
MDL锁主要作用是维护表元数据的数据一致性,在表上有活动事物的时候,不可以对元数据进行写入操作。
当对一张表进行增删改查操作时,加MDL读锁(共享)
当对表结构进行变更操作的时候,加MDL写锁(排他)
3.4.2 查看元数据锁
select * from performance_schema.metadata_locks;
3.5 意向锁
3.5.1 概念
主动添加
线程A对表进行添加表锁操作时,要检查表中是否有行级锁,非常麻烦,意向锁来解决
主动添加意向锁,线程A可以判断改表中的意向锁类型来判断是否可以加锁
3.5.2 分类
(1)意向共享锁(IS)
表锁共享锁(read)兼容
表锁排他锁(write)互斥
(2)意向排他锁 (IX)
表锁共享锁(read)互斥
表锁排他锁(write)互斥
意向锁之间不互斥
3.5.3 查看锁
select * from performance_schema.data_locks;
4 行级锁
4.1 概念
每次锁住对应的行数据。锁定颗粒度小,发生锁冲突的概率最低,并发度最高。应用在InnoDB存储引擎中。
InnoDB的数据时基于索引组织的,行锁时通过对索引上的索引项加锁来实现的,而不是对记录加的锁,行级锁分为三类:
(1)行锁(Record Lock)
锁定单个记录的锁,防止其他事物对此进行update和delete。在RC、RR隔离级别下都支持。
(2)间隙锁(Gap Lock)
锁定索引记录间隙(不含该记录)确保记录间隙不变,防止其他事物在这个间隙进行insert,产生幻读。在RR隔离级别下都支持
(3)临键锁 (Next-Key Lock)
行锁和间隙锁组合,同时锁住数据,并锁住数据前面的间隙Gap,在RR隔离级别下支持。
4.2 行锁
4.2.1 分类
(1)共享锁(S)
允许一个事物去读一行,阻止其他事务获得相同数据集的排他锁
(2)排他锁(X)
允许获取排他锁的事务更新数据,阻止其他事务获得相同数据集的共享锁和排他锁
4.2.2 加锁时机
4.2.3 查看锁
select
object_schema,
object_name,
index_name,
lock_type,
lock_mode,
lock_data
from
performance_schema.data_locks;
4.2.4 特点
(1)针对唯一索引进行检索时,对已存在的记录进行等值匹配时,将自动优化为行锁
(2)InnoDB的行锁针对索引加的锁,不通过索引条件检索数据,那么InnoDB将对表中的所有记录加锁,此时就会升级为表锁。
4.3 间隙锁&临键锁
(1)索引上的等值查询(唯一索引),给不存在的记录加锁是,优化为间隙锁
(2)索引上的等值查询(普通索引),向右遍历时最后一个值不满足查询需求时,next-key lock退化为间隙锁
(3)索引上的范围查询(唯一索引),会访问到不满足条件的第一个值为止