全局锁和表锁

了解MySQL的全局锁和表级锁mysql

全局锁

  • 对整个数据库实例加锁
  • 使用场景:作全库逻辑备份时,为了保证备份期间的库在同一个逻辑时间点,即一致性视图(相似于可重复读隔离级别的效果)

全局锁两种方式:sql

  • Flush tables with read lock(FTWRL) 使数据库处于只读状态,数据的增删改、数据定义语句和更新类事务的提交语句都会被阻塞
  • mysqldump 官方自带的逻辑备份工具,参数 -single-transaction 会在导数据以前启动一个事务,确保拿到一致性视图

以上两种方式不一样点:
使用mysqldump 前提是 引擎支持隔离级别 ,因此single-transaction方法只适用于支持事务引擎的库;MyISAM不支持事务,因此只能使用FTWRL命令数据库

set global readonly = true 
复制代码

这条语句也能够作到全库只读,可是不建议使用:安全

  • 有时readonly会用来判断库是主库仍是备库,所以修改global变量的方式影响会比较大
  • 异常处理方面:FTWRL后客户端异常断开,MySQL会自动释放全局锁,库恢复正常;设置readonly,客户端异常,则会保持readonly,会致使长时间处于不可写状态,风险较高

以上哪一种方式,一个库被全局上锁后,对立面任何一个表作字段操做,都会被锁住的多线程

即便没有全局加锁,有了表级锁,加字段也会遇到问题并发

表级锁

MySQL有两种表级别的锁:一种是表锁、另外一种是元数据锁(metadata lock,MDL)工具

表级锁的语法:lock tables xxx read/write

例如 线程A执行了lock table t1 read,t2 write 效果是 包括A线程在内的全部线程对于t1表只可读,写被阻塞;t2表读写都被阻塞
lock tables操做能够用unlock tables主动释放,也能够在客户端断开的时候自动释放。post

该操做不只阻塞其余线程的操做,也阻塞了当前线程的操做学习

对于innoDB这种支持行锁的引擎,通常不使用lock tables命令控制并发,影响过大spa

另外一种表级锁:MDL(metadata lock)

MDL在访问一个表的时候会自动加上,MDL的做用是,保证读写的正确性。当表作增删改查操做时,加MDL读锁;当对表结构变动的时候,加MDL写锁。

  • 读锁之间不互斥,多线程可对同一张表增删改查
  • 读写锁之间、写锁之间互斥。两个线程同时给一个表增长字段,则第二个须要等待第一个执行完才能继续
  • MDL锁在语句执行开始时申请,事务结束后释放

如何给小锁安全的加字段

  • 解决长事务,当作DDL变动的表中正好在执行长事务,则从information_schema库的innodb_trx找到当前执行长事务,先kill掉长事务或者暂停DDL
  • 比较理想的状态:修改表结构语句alter table能够设置等待时间,若是该时间内拿不到MDL锁,则该时间内拿不到MDL锁,则放弃执行,不会阻塞后面的语句
    MariaDB/ALiSQL已经支持 DDL NOWAIT/WAIT n 这个语法
alter table tb1_name NOWAIT add column...
alter table tb1_name WAIT N add column...   
复制代码

总结

对于使用innoDB引擎的库,建议使用–single-transaction 参数,对应用会更友好。


本文为极客时间《MySQL实战45讲》 的学习笔记,其中含有部分原文,若有侵权行为请联系我马上删除

第一节:一条SQL查询语句的执行过程

相关文章
相关标签/搜索