全局锁
加锁后,整个数据库都是只读状态,一般应用在数据库备份的时候,不能用其他事务的修改。但可以通过可重复读隔离级别通过开启事务创建ReadView来解决备份期间修改数据的情况。工具是mysqldump -single-transaction.
InnoDB存储引擎支持但MyiSAM引擎不支持。
表级锁
包括表锁 元数据锁 意向锁 AUTO-INC锁
表锁
对表加锁 共享锁(读锁)和独占锁(写锁)
表锁会限制别的线程读写,直到锁被释放,避免在InnoDB引擎的表使用,因为表锁颗粒度太大,影响并发性能。
元数据锁
对表进行CRUD加MDL读锁
对表进行结构变更操作加的是MDL写锁
阻塞情况:当有线程select时,自动加读锁,如果这时有其他线程要更改表结构(申请写锁)时,就会被阻塞,直到事务提交。同理先修改表结构后crud也会阻塞,但两个线程都是crud不会阻塞。
并且写锁获取优先级高于读锁,一旦出现写锁等待,会阻塞后面所有读锁。
意向锁
在对某些记录加共享锁前先加上意向共享锁
在对某些记录加独占锁前先加上意向独占锁
当执行插入更新删除操作时,先对表加意向独占锁再对记录加独占锁。
普通select不会加行级锁,通过MVCC实现一致性,但可以主动加锁。
表级锁和行级锁不发生冲突。
当加了意向锁,那么在加独占表锁时,先查有没有意向锁,如果有,就不用遍历表里记录了。
AUTO-INC锁
给某个字段声明AUTO-INCREMENT自增属性时,通过AUTO-INC锁实线。
锁在执行完插入语句后释放,一个事务持有AUTO-INC锁时候,别的事务插入会阻塞,在执行大量数据插入时影响插入性能,所以在MySQL5.1.22开始有了轻量级锁实现,轻量级锁在赋值后就释放。
innodb_autoinc_lock_mode
参数为0用AUTO-INC锁
参数为2用轻量级锁
参数为1混着用 能确定插入数量就轻量级 不确定用AUTO-INC锁,但在主从复制的场景是不安全的,虽然是性能最高的方式但因为并发插入的存在,导致自增的值不连续。
行级锁
记录锁 Record Lock
仅仅锁一条记录
间隙锁 Gap Lock
锁定一个范围,但是不包含记录本身
Next-Key Lock
上面两种锁的组合 锁定一个范围 也包括记录本身