群里一网友这两天刚入职新公司,遇到一个重启MySQL服务后,自动增长值丢失问题,差点背锅走人。下面我们一起来回顾一下这个问题。
在mysql中用自增列作为主键时,先往表里插入5条数据,此时表里数据id为1、2、3、4、5,如果此时删除id=4、5的数据后,再重启数据库,重启成功后向表里insert数据的时候,INNODB、MyISAM引擎下ID分别是从几开始增加?如果你没经历过,或者当面试时被问到这个问题时,相信多数人都是一脸懵逼。MD谁有事没事去重启线上数据库嘛。最主要的是很多没有测试过这个场景,没有这方面的经验,我在这里做个笔记,大家轻点喷!
MySQL通常使用的引擎都是INNODB,在建表时,一般使用自增列作为表的主键,这样的表对提高性能有一定的帮助。但是自增列有一个坑,并且这个坑存在了很久,一直到MySQL8.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表:自增列作为主键IDCREATETABLE`user`(`id`int(11)unsignedNOTNULLAUTO_INCREMENT,`name`varchar()NOTNULLDEFAULT,`age`int(4)unsignedNOTNULLDEFAULT0,PRIMARYKEY(`id`))ENGINE=InnoDBAUTO_INCREMENT=1DEFAULTCHARSET=utf8;//2.插入5条数据INSERTINTO`user`(`name`,age)VALUES(刘备1,21);INSERTINTO`user`(`name`,age)VALUES(刘备2,22);INSERTINTO`user`(`name`,age)VALUES(刘备3,23);INSERTINTO`user`(`name`,age)VALUES(刘备4,24);INSERTINTO`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引擎表中的自增列不会存在这个问题。
MySQL8.0auto-increment计数器逻辑
在MySQL8.0中,这个计数器的逻辑变了:每当计数器的值有变,InnoDB会将其写入redolog,保存到引擎专用的系统表中。MySQL正常关闭后重启:从系统表中获取计数器的值。MySQL故障后重启:从系统表中获取计数器的值;从最后一个检查点开始扫描redolog中记录的计数器值;取这两者的最大值作为新值。
总结
1)如果mysql重启了,那么innodb表在启动后,AUTO_INCREMENT值会自动检测出、并重置为当前表中自增列的最大值+1。2)假如一个表里AUTO_INCREMENT计数器的值是10,此时执行updatetablesetid=15whereid=9后,如果这时再继续插入数据,到了自增ID=15的时候是会报错。
但是这个时候继续插入,就不会报错。因为刚才即使报错了,AUTO_INCREMENT的值依旧会增加。3)现在使用的一般都是innodb引擎,如果将myisam引擎转换过来的时候,一定要小心这个引擎在自增id上的不同表现。在主从使用不同引擎的时候,也会出现问题,最好将引擎改完一致性的。