群里一网友这两天刚入职新公司,遇到一个重启 MySQL 服务后,自动增加值丢失问题,差点背锅走人。下面咱们一块儿来回顾一下这个问题。java
在 mysql 中用自增列做为主键时,先往表里插入 5 条数据,此时表里数据 id 为 一、二、三、四、5,若是此时删除 id=四、5 的数据后,再重启数据库,重启成功后向表里 insert 数据的时候,INNODB、MyISAM 引擎下 ID 分别是从几开始增长?若是你没经历过,或者当面试时被问到这个问题时,相信多数人都是一脸懵逼。MD 谁有事没事去重启线上数据库嘛。最主要的是不少没有测试过这个场景,没有这方面的经验,我在这里作个笔记,你们轻喷!mysql
MySQL 一般使用的引擎都是 INNODB,在建表时,通常使用自增列做为表的主键,这样的表对提升性能有必定的帮助。可是自增列有一个坑,而且这个坑存在了好久,一直到 MySQL 8.0 版本,才修复了这个坑,这个坑就是表的自增列变量 auto\_increment 在 MySQL 重启后,有可能丢失。面试
在 MySQL 低版本中,InnoDB 表中使用自增的 auto-increment 计数器 会把值存放在内存中,不会写入磁盘。一旦 MySQL 服务重启,这个值就丢了,InnoDB 引擎会根据表中现有的数据从新计算该计数器的值:获取表中最大的自增主键 ID 做为auto-increment 计数器的最大计数,当 insert 数据时,在 auto-increment 计数器最大值上 1。sql
先建立一张 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 为 四、5 的数据,再向 user 表中插入一条数据,主键 ID 是 auto-increment 的值 6。测试
** mysql 数据库重启后,innodb 自增主键 ID 会根据 auto-increment 计数器的重置而重置。**spa
在场景一的基础上,在删除 id 为 六、3 的数据后,此时 auto-increment 计数器的值为 7,user 表里的 id 最大是 2。code
而后重启数据库后,auto-increment 计数器的值变为 3,也就是 user 表里的自增列 ID 的最大值 2 加 1。
此时在插入数据时,自增 ID 会从 3 开始自增。Innodb 表中把自增列做为主键 ID 时,在 mysql 重启后就会存在 ID 重置问题。**删除数据后,再重启,AUTO\_INCREMENT 会查询表里最大 ID 并进行重置,重置后和重启前AUTO\_INCREMENT 计数器的值不一样。**在 MyISAM 引擎表中的自增列不会存在这个问题。
在 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 上的不一样表现。在主从使用不一样引擎的时候,也会出现问题,最好将引擎改完一致性的。