引言

  根据加锁的范围,MySQL 里面的锁大致可以分成全局锁、表级锁和行锁三类。这篇文章,与你分享全局锁和表级锁。而关于行锁的内容,我会留着在下一篇文章中再和你详细介绍。

全局锁

  全局锁就是对整个数据库实例加锁。MySQL 提供了一个加全局读锁的方法,命令是 Flush tables with read lock(FTWRL)。当你需要让整个库处于只读状态的时候,可以使用这个命令,之后其他线程的以下语句会被阻塞:数据更新语句(数据的增删改)、数据定义语句(包括建表、修改表结构等)和更新类事务的提交语句。
  全局锁的典型使用场景是,做全库逻辑备份。也就是把整库每个表都 select 出来存成文本。
  官方自带的逻辑备份工具是 mysqldump。当 mysqldump 使用参数–single-transaction的时候,导数据之前就会启动一个事务,来确保拿到一致性视图。而由于MVCC 的支持,这个过程中数据是可以正常更新的。
  你也许会问,既然要全库只读,为什么不使用 set global readonly=true 的方式呢?确实 readonly 方式也可以让全库进入只读状态,但我还是会建议你用 FTWRL 方式,主要有两个原因:

  1. 在有些系统中,readonly 的值会被用来做其他逻辑,比如用来判断一个库是主库还是备库。因此,修改 global 变量的方式影响面更大,我不建议你使用。
  2. 在异常处理机制上有差异。如果执行 FTWRL 命令之后由于客户端发生异常断开,那么 MySQL 会自动释放这个全局锁,整个库回到可以正常更新的状态。而将整个库设置为 readonly 之后,如果客户端发生异常,则数据库就会一直保持 readonly 状态,这样会导致整个库长时间处于不可写状态,风险较高。

表级锁

  MySQL 里面表级别的锁有两种:一种是表锁,一种是元数据锁(meta data lock,MDL)。
  表锁的语法是 lock tables … read/write。与 FTWRL 类似,可以用 unlock tables 主动释放锁,也可以在客户端断开的时候自动释放。需要注意,lock tables 语法除了会限制别的线程的读写外,也限定了本线程接下来的操作对象。
  举个例子, 如果在某个线程 A 中执行 lock tables t1 read, t2 write; 这个语句,则其他线程写 t1、读写 t2 的语句都会被阻塞。同时,线程 A 在执行 unlock tables 之前,也只能执行读 t1、读写 t2 的操作。连写 t1 都不允许,自然也不能访问其他表。
  另一类表级的锁是 MDL(metadata lock)。MDL 不需要显式使用,在访问一个表的时候会被自动加上。MDL 的作用是,保证读写的正确性。你可以想象一下,如果一个查询正在遍历一个表中的数据,而执行期间另一个线程对这个表结构做变更,删了一列,那么查询线程拿到的结果跟表结构对不上,肯定是不行的。
  因此,在 MySQL 5.5 版本中引入了 MDL,当对一个表做增删改查操作的时候,加 MDL读锁;当要对表做结构变更操作的时候,加 MDL 写锁。

  • 读锁之间不互斥,因此你可以有多个线程同时对一张表增删改查。读写锁之间、
  • 写锁之间是互斥的,用来保证变更表结构操作的安全性。
    因此,如果有两个线程要同时给一个表加字段,其中一个要等另一个执行完才能开始执行。

  虽然 MDL 锁是系统默认会加的,但却是你不能忽略的一个机制。比如下面这个例子,我经常看到有人掉到这个坑里:给一个小表加个字段,导致整个库挂了。
  你肯定知道,给一个表加字段,或者修改字段,或者加索引,需要扫描全表的数据。在对大表操作的时候,你肯定会特别小心,以免对线上服务造成影响。而实际上,即使是小表,操作不慎也会出问题。
我们来看一下下面的操作序列,假设表 t 是一个小表。

  1. 我们可以看到 session A 先启动,这时候会对表 t 加一个 MDL 读锁。由于 session B 需要的也是 MDL 读锁,因此可以正常执行。
  2. 之后 session C 会被 blocked,是因为 session A 的 MDL 读锁还没有释放,而 session C需要 MDL 写锁,因此只能被阻塞。
  3. 如果只有 session C 自己被阻塞还没什么关系,但是之后所有要在表 t 上新申请 MDL 读锁的请求也会被 session C 阻塞。
  4. 前面我们说了,所有对表的增删改查操作都需要先申请MDL 读锁,就都被锁住,等于这个表现在完全不可读写了。如果某个表上的查询语句频繁,而且客户端有重试机制,也就是说超时后会再起一个新session 再请求的话,这个库的线程很快就会爆满。

  基于上面的分析,我们来讨论一个问题,如何安全地给小表加字段?
  首先我们要解决长事务,事务不提交,就会一直占着 MDL 锁。在 MySQL 的information_schema 库的 innodb_trx 表中,你可以查到当前执行中的事务。如果你要做DDL 变更的表刚好有长事务在执行,要考虑先暂停 DDL,或者 kill 掉这长事务。
  但考虑一下这个场景。如果你要变更的表是一个热点表,虽然数据量不大,但是上面的请求很频繁,而你不得不加个字段,你该怎么做呢?
  这时候 kill 可能未必管用,因为新的请求马上就来了。比较理想的机制是,在 alter table语句里面设定等待时间,如果在这个指定的等待时间里面能够拿到 MDL 写锁最好,拿不到也不要阻塞后面的业务语句,先放弃。之后开发人员或者 DBA 再通过重试命令重复这个过程。

06 | 全局锁和表锁 : 给表加个字段怎么有这么多阻碍相关推荐

  1. 06 全局锁和表锁 :给表加个字段怎么有这么多阻碍? *(笔记)转

    数据库锁设计的初衷是处理并发问题.作为多用户共享的资源,当出现并发访问的时候,数据库需要合理地控制资源的访问规则.而锁就是用来实现这些访问规则的重要数据结构. 根据加锁的范围,MySQL 里面的锁大致 ...

  2. 六、全局锁和表锁 :给表加个字段怎么有这么 多阻碍?

    提示:写完文章后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 前言 1 全局锁 2 表级锁 2.1 表锁 2.2 元数据锁 前言 接下来聊聊 MySQL 的锁.数据库锁设计的初衷是处理并 ...

  3. mysql BDB支持表锁吗_mysql 表锁问题

    本文转自:http://www.cnblogs.com/itdragon/p/8194622.html MySQL 表锁和行锁机制 行锁变表锁,是福还是坑?如果你不清楚MySQL加锁的原理,你会被它整 ...

  4. MySQL锁:全局锁、表级锁和行锁

    事务的实现离不开MySQL数据库的锁机制,设计锁的目的也是为了处理并发访问问题,本文简单介绍MySQL 里面的全局锁.表级锁和行锁三类锁. 目录 全局锁 表级锁 表锁 1.读锁实例 2.写锁实例 元数 ...

  5. mysql 写锁需要等待读锁释放吗_Mysql实战45讲笔记:5、全局锁和表锁

    全局锁: 对整个数据库实例加锁. MySQL提供加全局读锁的方法:Flush tables with read lock(FTWRL) 这个命令可以使整个库处于只读状态.使用该命令之后,数据更新语句. ...

  6. mysql怎么加全局锁_MySQL锁机制/管理(并发锁,行锁,表锁,预加锁,全局锁等等)

    MySQL实验室 1.?MySQL 中并发和隔离控制机制 Meta-data元数据锁:在table cache缓存里实现的,为DDL(Data Definition Language)提供隔离操作.一 ...

  7. MySQL 全局锁和表锁

    日常读书笔记 全局锁 含义和命令行 典型使用场景 注意点 问题:备份一定要加锁吗? 问题:既然备份不一定要加锁,那全局锁存在的意义是什么? 表级锁 表锁 命令行 所释放时机 锁时的具体细节 元数据锁 ...

  8. mysql全局读写怎么锁_MySQL全局锁和表锁

    最近正好在看操作系统中关于线程,锁的部分,也可以学习一下数据库种的锁. MySQL的锁分为全局锁,表级锁,行锁三类. 全局锁 名思义,全局锁就是对整个数据库实例加锁.MySQL提供了一个加全局读锁的方 ...

  9. MySQL - 全局锁、表级锁、行级锁、元数据锁、自增锁、意向锁、共享锁、独占锁、记录锁、间隙锁、临键锁、死锁

    # 前言 本篇只介绍 MySQL 锁的基本知识. 我的 MySQL 版本是 MySQL 5.7.34, 建议使用 MySQL 5.6 及之后的版本. ## 先上一个图 ## 为什么要使用锁? 个人理解 ...

最新文章

  1. 【OpenCV 4开发详解】直方图操作
  2. 基于Python的频谱分析(一)
  3. 用耳朵“打字”| 这个设备可以让患者实现用耳朵进行交流
  4. freebsd 下安装桌面
  5. js轮播图片小圆点变化_原生js实现轮播图(两种方法)
  6. Python中操作mysql知识(一)
  7. 【引用】jQuery 选择器
  8. jdk安装包_第一章(第1节):安装JDK
  9. 自制时间比对函数处理 比对过去时间与当前时间相差多少年多少月多少周多少分 多少秒...
  10. mybatis mysql 事务处理_详解Java的MyBatis框架中的事务处理
  11. httperf ---linux web站点压力测试
  12. 程序员是制造 Bug 的“元凶”?
  13. bzoj3067: Hyperdrome
  14. java读取txt生成excel_读取TXT文件内容,生成Excel文件
  15. 用云真机测试本地应用程序
  16. 点餐小程序开发(软件源码)
  17. html+css发光字体
  18. 银联Pos终端签到、签退、批结算、批上送、PinKey、MacKey、KEK、主密钥、工作密钥、TPDU、报文头
  19. 【服务器】nvidia驱动重装
  20. 14 最小二乘估计原理推导和线性回归的外推等

热门文章

  1. matlab语言中的assert断言函数
  2. C专家编程--读书笔记十 再论指针
  3. http://www.cnblogs.com/Bear-Study-Hard/archive/2008/03/26/1123267.html
  4. 基于matlab的智能天线波束方向图仿真,基于MATLAB的智能天线波束方向图仿真
  5. linux lpte_linux常用命令
  6. 小米6 twrp_小米6刷上统信 UOS 国产系统,操作流畅但安装需谨慎!
  7. python xlsx 图片_实例11:用Python给Excel所有工作表添加图片
  8. pptx和ppt计算机二级,计算机二级Ms office考试PPT考点总结.pptx
  9. 云服务器网站不能够上传视频,网站的视频要存到云服务器上吗
  10. curl 慢 不稳定_Node.js VM 不完全指北