关于java:好险一入职就遇到MySQL这么大Bug差点背锅走人

38次阅读

共计 2387 个字符,预计需要花费 6 分钟才能阅读完成。

送大家以下 java 学习材料





群里一网友这两天刚入职新公司,遇到一个重启 MySQL 服务后,主动增长值失落问题,差点背锅走人。上面咱们一起来回顾一下这个问题。

在 mysql 中用自增列作为主键时,先往表里插入 5 条数据,此时表里数据 id 为 1、2、3、4、5,如果此时删除 id=4、5 的数据后,再重启数据库,重启胜利后向表里 insert 数据的时候,INNODB、MyISAM 引擎下 ID 别离是从几开始减少?如果你没经验过,或者当面试时被问到这个问题时,置信少数人都是一脸懵逼。MD 谁有事没事去重启线上数据库嘛。最次要的是很多没有测试过这个场景,没有这方面的教训,我在这里做个笔记,大家轻喷!

MySQL 通常应用的引擎都是 INNODB,在建表时,个别应用自增列作为表的主键,这样的表对进步性能有肯定的帮忙。然而自增列有一个坑,并且这个坑存在了很久,始终到 MySQL 8.0 版本,才修复了这个坑,这个坑就是表的自增列变量 auto\_increment 在 MySQL 重启后,有可能失落。

  • 「innodb 引擎(低版本):Innodb 表中把自增列作为主键 ID 时,自增列是通过 auto-increment 计数器实现的,计数器的最大值是记录到内存中的,重启数据库后,会导致 auto-increment 计数器重置,从而会导致主键 ID 重置。」
  • 「MyISam 引擎:MyISAM 表会把自增列 (auto-increment 计数器) 最大值是记录到数据文件里,重启 MySQL 自增列 (计数器) 最大值不会失落,从而应用自增列作为主键 ID 时也不会失落。」

innodb 主键重置问题

在 MySQL 低版本中,InnoDB 表中应用自增的 auto-increment 计数器 会把值寄存在内存中,不会写入磁盘。一旦 MySQL 服务重启,这个值就丢了,InnoDB 引擎会依据表中现有的数据从新计算该计数器的值:获取表中最大的自增主键 ID 作为 auto-increment 计数器的最大计数,当 insert 数据时,在 auto-increment 计数器最大值上 1。

先创立一张 user 表,新增几条数据:

`//1. 创立 user 表:自增列作为主键 ID`
``CREATE TABLE `user` (``
 `` `id` int(11) unsigned NOT NULL AUTO_INCREMENT,``
 `` `name` varchar(255) NOT NULL DEFAULT '',``
 `` `age` int(4) unsigned NOT NULL DEFAULT '0',``
 ``PRIMARY KEY (`id`)``
`) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8;`
`//2. 插入 5 条数据 `
``INSERT INTO `user`(`name`, age) VALUES('刘备 1', 21);``
``INSERT INTO `user`(`name`, age) VALUES('刘备 2', 22);``
``INSERT INTO `user`(`name`, age) VALUES('刘备 3', 23);``
``INSERT INTO `user`(`name`, age) VALUES('刘备 4', 24);``
``INSERT INTO `user`(`name`, age) VALUES('刘备 5', 25);``

场景一

「mysql 数据库不重启时,innodb 自增主键 ID 会依据 auto-increment 计数器始终递增。」

向 user 表里插入 5 条数据,主键 ID 按自增列通过 auto-increment 计数器实现自增。

在 user 表里删除 id 为 4、5 的数据,再向 user 表中插入一条数据,主键 ID 是 auto-increment 的值 6。

场景二

** mysql 数据库重启后,innodb 自增主键 ID 会依据 auto-increment 计数器的重置而重置。**

在场景一的根底上,在删除 id 为 6、3 的数据后,此时 auto-increment 计数器的值为 7,user 表里的 id 最大是 2。

而后重启数据库后,auto-increment 计数器的值变为 3,也就是 user 表里的自增列 ID 的最大值 2 加 1。

此时在插入数据时,自增 ID 会从 3 开始自增。Innodb 表中把自增列作为主键 ID 时,在 mysql 重启后就会存在 ID 重置问题。** 删除数据后,再重启,AUTO\_INCREMENT 会查问表里最大 ID 并进行重置,重置后和重启前 AUTO\_INCREMENT 计数器的值不同。** 在 MyISAM 引擎表中的自增列不会存在这个问题。

MySQL 8.0 auto-increment 计数器逻辑

在 MySQL 8.0 中,这个计数器的逻辑变了:每当计数器的值有变,InnoDB 会将其写入 redo log,保留到引擎专用的零碎表中。MySQL 失常敞开后重启:从零碎表中获取计数器的值。MySQL 故障后重启:从零碎表中获取计数器的值;从最初一个检查点开始扫描 redo log 中记录的计数器值;取这两者的最大值作为新值。

总结

1)如果 mysql 重启了,那么 innodb 表在启动后,AUTO\_INCREMENT 值会自动检测出、并重置为以后表中自增列的最大值 +1。2)如果一个表里 AUTO\_INCREMENT 计数器的值是 10,此时执行 update table set id = 15 where id = 9 后,如果这时再持续插入数据,到了自增 ID=15 的时候是会报错。然而这个时候持续插入,就不会报错。因为方才即便报错了,AUTO\_INCREMENT 的值依旧会减少。3)当初应用的个别都是 innodb 引擎,如果将 myisam 引擎转换过去的时候,肯定要小心这个引擎在自增 id 上的不同体现。在主从应用不同引擎的时候, 也会呈现问题,最好将引擎改完一致性的。

正文完
 0