data:image/s3,"s3://crabby-images/02f21/02f21620e18cd8f93014eaaf35e42e1b3a4bcd48" alt=""
问
当我们使用一个事务操作很多数据时, MySQL 有时会报错: The total number of locks exceeds the lock table size
根据官方文档, 我们需要调大 buffer pool 的大小:
data:image/s3,"s3://crabby-images/d1a93/d1a939997d3a1cbf591bb7694e31659b7a1079bb" alt=""
本期实验, 我们来探索一下锁用得多与 buffer pool 大小的关系
实验
我们用老方法建一个数据库, 并将 buffer pool 大小调整到了最小值5M, 方便我们复现问题
data:image/s3,"s3://crabby-images/b0107/b0107c1483c5d4bc64675f5f26725b9feab5ccb4" alt=""
现在来模拟一个用锁特别多的事务:
data:image/s3,"s3://crabby-images/183e6/183e666eb8d2ed0631278ab08755bb82d1a8e371" alt=""
我们还是用老方法让表翻倍, 来不停地占用锁.
看一下效果:
data:image/s3,"s3://crabby-images/fd9ce/fd9ce9ec5ab88f0d2b912303fa079276ad24db61" alt=""
我们可以通过 information_schema.INNODB_TRX 来查看事务使用了多少锁, 解释一下上图中标记的这几个状态:
-
trx_tables_locked: 该事务锁了几张表 -
trx_rows_locked: 该事务锁了多少数据行 -
trx_lock_structs: 该事务一共用到了多少个锁结构.一个锁结构用于锁住多个表或多个行 -
trx_lock_memory_bytes: 该事务的锁结构一共用了多少内存
再来看看 buffer pool 的状态:
data:image/s3,"s3://crabby-images/1c7de/1c7deabd0cfd9b071a236dfb562fe6b594551a7a" alt=""
解释一下 Buffer pool 的这两个状态:
-
total 是 Buffer pool 的总页数 -
misc 是 Buffer pool 中非数据页的页数
我们继续造数据, 让该事务使用的锁越来越多, 再来看看状态:
data:image/s3,"s3://crabby-images/9f2ae/9f2aef6567d83d94b267029109b07aa3c96b6334" alt=""
与最初的状态相比, 该事务使用的锁的内存增长了 (1269968 – 24784 = ) 1245184 字节 = 1216 k, 而 buffer pool 非数据页多使用了 (84-8 = ) 76页, 每页16k, 总共 1216 k
也就是说, 该事务的锁内存均分自 buffer pool .
我们继续造数据, 造到报错为止, 不停查看状态:
data:image/s3,"s3://crabby-images/09183/091835e0cb9feaee495ba68a4854478db24d42ae" alt=""
data:image/s3,"s3://crabby-images/096ef/096ef1090828cbf188d9d291e92638c384a0ca06" alt=""
可以看到, 发生报错时, 事务会回滚. 回滚前, 该事务的锁占用的内存大小是 3842256 字节, 也就是 ( 3842256 / 1024 / 16 = ) 234.5 页, 此时 buffer pool 使用率大概是 ( 234.5 / 320 = ) 73%
我们的实验到此结束, 通过此实验我们知道:
-
如果事务用到了很多锁, 那么锁结构使用的内存是从 Buffer pool 中分配的 -
Buffer pool 使用率比较高时, 就会报错: The total number of locks exceeds the lock table size
关于 Buffer pool 使用到多少, 会进行报错, 我们参考如下 MySQL 源码:
data:image/s3,"s3://crabby-images/0c363/0c363e3d846c7bcdf7c455fa71bad0fb8cdfd5b5" alt=""
我们可以看到这个理论阈值是75%, 当 Buffer pool 中 非数据页的大小超过了75%, 我们就会再无法分配更多的锁.
(实验中测定的73%与这个理论阈值相差不大)
小贴士
当 Buffer pool 的大小不足时, Buffer pool 可以回收数据页 (干净页可直接回收, 脏页可刷脏后回收), 而不能直接回收 非数据页.
MySQL 设定这个75%的阈值, 也是为了让 Buffer pool 中留一部分空间给数据页, 毕竟 Buffer pool 的最大作用是给数据页做缓存.
关于 MySQL 的技术内容,你们还有什么想知道的吗?赶紧留言告诉小编吧!
data:image/s3,"s3://crabby-images/4fe4f/4fe4f390a23eb28cadcc0a57109e0e60a9009d4e" alt=""