为什么MySQL8新特性会修改自增主键属性


Posted in MySQL onApril 18, 2022

一、简述

MySQL版本从5直接大跃进到8,相信MySQL8一定会有很多令人意想不到的改进,如果不想只会CRUD可以看看。

比如系统表引擎的变化-全部换成事务型的InnoDB。

MySQL5.7系统部引擎

为什么MySQL8新特性会修改自增主键属性

MySQL8系统引擎

为什么MySQL8新特性会修改自增主键属性

上图可以看到,MySQL5.7的系统表引擎有MEMORY、InnnoDB和MyISAM三种,但MySQL8的系统表引擎都换成了InnoDB。MySQL8新特性还有很多,接下来进入正题康康它的自增主键。

二、MySQL自增主键

为什么MySQL8新特性会修改自增主键属性?

在MySQL8.0之前,自增主键 AUTO_INCREMENT 的值如果大于max(primary key) +1,那么在MySQL重启后,则会重置 AUTO_INCREMENT = max(primary key)+1 的值,这种现象在某些情况下会导致业务主键冲突或者其他难以发现的一些问题。

MySQL官网解释自增ID冲突问题

为什么MySQL8新特性会修改自增主键属性

因为在MySQL5.7中,对于自增主键的分配规则是由InnoDB数据字典内部一个计数器来决定的,而该计数器维护在了内存中,并不会持久化到磁盘中,此时硬盘中并无数据,当数据库重启的时候,该计数器会被初始化为: auto_increment = max(primary key)+1。

如何解决自增主键冲突问题?

这个问题一直到MySQL8.0才解决。

8.0版本将会对 AUTO_INCREMENT 值进行持久化,所以即使MySQL重启后该值也不会改变。

即其将自增主键的计数器持久化到了重做日志中,每次计数器发生改变都会将其写入到重做日志中,如果这个时候数据库重启了,那么InnoDB数据字典会根据重做日志中的信息来初始化计数器的内存值,就可以恢复到了上次关闭数据库前的状态,通过自增ID持久化来避免8.0之前可能会出现的问题。

三、自增主键测试

分别在MySQL5和MySQL8上进行自增主键测试。

1、MySQL5.7自增主键

在MySQL5.7中的,这里我们先创建一个数据表,这个数据表中设置一个自增列。

CREATE TABLE t_test_auto_increment_tjt(
  `id` bigint(20) NOT NULL AUTO_INCREMENT COMMENT '主键ID',
  `test_key` varchar(50) NOT NULL COMMENT '名称',
  `test_value` varchar(50) DEFAULT NULL COMMENT '值',
  PRIMARY KEY (`id`)
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8mb4 COMMENT='测试主键自增表';

然后向自增主键表中添加了4条记录,表中的四条添加的记录的id字段值就分别为: 1、2、3、4。

INSERT INTO t_test_auto_increment_tjt(id, test_key, test_value) VALUES
('0','吞噬星空','停更'),
('0','水斗大陆','可以停播了'),
('0','武神主宰','装B还得看尘少'),
('0','完美世界','yyds')

插入数据的SQL添加的是0,其实就是默认赋值,表 t_test_auto_increment_tjt 中的自增列是不可以添加0或者null的,那么这个时候表中的四条添加的记录的id字段值就分别为: 11、2、3、4。

为什么MySQL8新特性会修改自增主键属性

接下来,将表中的id为4的字段删除。

DELETE FROM t_test_auto_increment_tjt WHERE id = 4

然后,继续在表中添加一条记录,执行之后我们可以发现,此时自增主键的ID结果是5。

INSERT INTO t_test_auto_increment_tjt(id, test_key, test_value) VALUES ('0','完美世界','yyds-YYDS')

为什么MySQL8新特性会修改自增主键属性

因为我们前面已经将表中id为4的记录删除了,这个时候下一次自增的时候即使表中没有id为4的字段了,但是这个时候我们也不会添加4,而是添加5。其实这个时候就是自增主键的值auto_increment 大于了max(primary key)+1。

再接下来,将表中的id为5的记录删除。

DELETE FROM t_test_auto_increment_tjt WHERE id = 5

最后,重启MySQL数据库,再向表中添加一条记录。

INSERT INTO t_test_auto_increment_tjt(id, test_key, test_value) VALUES ('0','完美世界','yyds-YYDS-restart')

为什么MySQL8新特性会修改自增主键属性

上图可以看到,重启后 重启后 重启后 执行的结果中添加的记录的id值为 : 4, 按之前的操作来看4和5已经被删除了,那么添加的就应该是6,为什么是4呢?

因为在MySQL5.7中,自增主键的分配规则是由InnoDB数据字典内部一个计数器来决定的,而该计数器维护在了内存中,并不会持久化到磁盘中,此时硬盘中并无数据,当数据库重启之后该计数器会被初始化为: auto_increment = max(primary key)+1,所以记录的id=4,而不是6。

2、MySQL8自增主键

在MySQL8中,按照上述MySQL5.7的操作步骤测试自增主键问题。

首先创建自增主键表、插入数据。

为什么MySQL8新特性会修改自增主键属性

然后,删除数据、插入数据。

为什么MySQL8新特性会修改自增主键属性

最后,重启 重启 重启 重启后插入数据。

为什么MySQL8新特性会修改自增主键属性

一定要彻底关闭MySQL服务,然后重新启动。

为什么MySQL8新特性会修改自增主键属性

重启后插入数据,测试自增主键ID的值?

为什么MySQL8新特性会修改自增主键属性

总结

到此这篇关于MySQL8自增主键变化的文章就介绍到这了!

MySQL 相关文章推荐
mysql优化
Apr 06 MySQL
一篇文章弄懂MySQL查询语句的执行过程
May 07 MySQL
MySQL中VARCHAR与CHAR格式数据的区别
May 26 MySQL
MySQL查看表和清空表的常用命令总结
May 26 MySQL
如何自己动手写SQL执行引擎
Jun 02 MySQL
浅谈MySQL表空间回收的正确姿势
Oct 05 MySQL
Linux7.6二进制安装Mysql8.0.27详细操作步骤
Nov 27 MySQL
SQL注入详解及防范方法
Dec 06 MySQL
彻底解决MySQL使用中文乱码的方法
Jan 22 MySQL
Innodb存储引擎中的后台线程详解
Apr 03 MySQL
MySQL分区路径子分区再分区
Apr 13 MySQL
MySQL的存储过程和相关函数
Apr 26 MySQL
SQL语句多表联合查询的方法示例
Apr 18 #MySQL
Mysql 如何合理地统计一个数据库里的所有表的数据量
Apr 18 #MySQL
MySQL三种方式实现递归查询
Apr 18 #MySQL
Mysql 8.x 创建用户以及授予权限的操作记录
Golang连接并操作MySQL
Apr 14 #MySQL
以MySQL5.7为例了解一下执行计划
Apr 13 #MySQL
MySQL时区造成时差问题
You might like
PHP中创建并处理图象
2006/10/09 PHP
PHP控制网页过期时间的代码
2008/09/28 PHP
使用php计算排列组合的方法
2013/11/13 PHP
使用PHP强制下载PDF文件示例
2014/01/17 PHP
ThinkPHP中I(),U(),$this->post()等函数用法
2014/11/22 PHP
PHP8.0新功能之Match表达式的使用
2020/07/19 PHP
用JS实现的一个include函数
2007/07/21 Javascript
jQuery用unbind方法去掉hover事件及其他方法介绍
2013/03/18 Javascript
通过JQuery实现win8一样酷炫的动态磁贴效果(示例代码)
2013/07/13 Javascript
利用jQuery实现可以编辑的表格
2014/05/26 Javascript
完美兼容各大浏览器的jQuery仿新浪图文淡入淡出间歇滚动特效
2014/11/12 Javascript
JS实现动态移动层及拖动浮层关闭的方法
2015/04/30 Javascript
JS百度地图搜索悬浮窗功能
2017/01/12 Javascript
Node.js和Express简单入门介绍
2017/03/24 Javascript
动态统计当前输入内容的字节、字符数的实例详解
2017/10/27 Javascript
JS排序算法之冒泡排序,选择排序与插入排序实例分析
2017/12/13 Javascript
详解Vue 如何监听Array的变化
2019/06/06 Javascript
Vue页面手动刷新,实现导航栏激活项还原到初始状态
2020/08/06 Javascript
[01:22]DOTA2神秘商店携大量周边降临完美大师赛
2017/11/07 DOTA
Python的装饰器使用详解
2017/06/26 Python
Python操作mongodb数据库进行模糊查询操作示例
2018/06/09 Python
Python使用pymysql从MySQL数据库中读出数据的方法
2018/07/25 Python
python 获取键盘输入,同时有超时的功能示例
2018/11/13 Python
Python使用sqlalchemy模块连接数据库操作示例
2019/03/13 Python
python开头的coding设置方法
2019/08/08 Python
浅谈pytorch卷积核大小的设置对全连接神经元的影响
2020/01/10 Python
三步解决python PermissionError: [WinError 5]拒绝访问的情况
2020/04/22 Python
如何对python的字典进行排序
2020/06/19 Python
关于python tushare Tkinter构建的简单股票可视化查询系统(Beta v0.13)
2020/10/19 Python
全面介绍python中很常用的单元测试框架unitest
2020/12/14 Python
Myprotein丹麦官网:欧洲第一运动营养品牌
2019/04/15 全球购物
北京-环亚运商测试题.net程序员初步测试题
2013/05/28 面试题
服装行业创业计划书范文
2014/02/05 职场文书
进步之星获奖感言
2014/02/22 职场文书
机关干部三严三实心得体会
2014/10/13 职场文书
详解Python 3.10 中的新功能和变化
2021/04/28 Python