为什么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 05 MySQL
MySQL慢查询的坑
Apr 28 MySQL
MySQL查询学习之基础查询操作
May 08 MySQL
MySQL中InnoDB存储引擎的锁的基本使用教程
May 26 MySQL
一文读懂navicat for mysql基础知识
May 31 MySQL
MySQL库表名大小写的选择
Jun 05 MySQL
MySQL高速缓存启动方法及参数详解(query_cache_size)
Jul 01 MySQL
MySQL系列之十四 MySQL的高可用实现
Jul 02 MySQL
mysql5.7的安装及Navicate长久免费使用的实现过程
Nov 17 MySQL
深入讲解数据库中Decimal类型的使用以及实现方法
Feb 15 MySQL
MySql数据库触发器使用教程
Jun 01 MySQL
MySQL中TIMESTAMP类型返回日期时间数据中带有T的解决
Dec 24 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 INI配置文件的解析实现分析
2011/01/04 PHP
通过修改Laravel Auth使用salt和password进行认证用户详解
2017/08/17 PHP
js下写一个事件队列操作函数
2010/07/19 Javascript
单独使用CKFinder选择图片的方法
2010/08/21 Javascript
javascript 学习笔记(一)DOM基本操作
2011/04/08 Javascript
javascript实现json页面分页实例代码
2014/02/20 Javascript
php中给js数组赋值方法
2014/03/10 Javascript
js获取json元素数量的方法
2015/01/27 Javascript
js实现动画特效的文字链接鼠标悬停提示的方法
2015/03/02 Javascript
前端js文件合并的三种方式推荐
2016/05/19 Javascript
vue结合Echarts实现点击高亮效果的示例
2018/03/17 Javascript
Python搭建HTTP服务器和FTP服务器
2017/03/09 Python
Python中正则表达式详解
2017/05/17 Python
python根据unicode判断语言类型实例代码
2018/01/17 Python
使用python批量化音乐文件格式转换的实例
2019/01/09 Python
vue学习笔记之动态组件和v-once指令简单示例
2020/02/29 Python
python3列表删除大量重复元素remove()方法的问题详解
2021/01/04 Python
澳大利亚最大的网上油画销售画廊:Direct Art Australia
2018/04/15 全球购物
Camper鞋西班牙官方网上商店:西班牙马略卡岛的鞋类品牌
2019/03/14 全球购物
东南亚冒险旅行与活动:Adventoro
2019/10/16 全球购物
网上常见的一份Linux面试题(多项选择部分)
2015/02/07 面试题
高中毕业的自我鉴定
2013/12/09 职场文书
四年大学生活的个人自我评价
2013/12/11 职场文书
幼儿园校车司机的岗位职责
2014/01/30 职场文书
社区安全检查制度
2014/02/03 职场文书
2014年政风行风自查自纠报告
2014/10/21 职场文书
2014年打非治违工作总结
2014/11/13 职场文书
2014年志愿者工作总结
2014/11/20 职场文书
庐山导游词
2015/02/03 职场文书
2015年保安个人工作总结
2015/04/02 职场文书
面试通知单大全
2015/04/20 职场文书
社区党员干部承诺书
2015/05/04 职场文书
情人节单身感言
2015/08/03 职场文书
初中英语教学反思范文
2016/02/15 职场文书
MySQL系列之十 MySQL事务隔离实现并发控制
2021/07/02 MySQL
详解Python flask的前后端交互
2022/03/31 Python