Mysql中自增主键是如何工作的

2023-11-23 15:20

本文主要是介绍Mysql中自增主键是如何工作的,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

自增主键的特点是当表中每新增一条记录时,主键值会根据自增步长自动叠加,通常会将自增步长设置1,也就是说自增主键值是连续的。那么MySQL自增主键值一定会连续吗?今天这篇文章就来说说这个问题,看看什么情况下自增主键会出现不连续?

1.数据准备

drop TABLE increnment_test;
-- 创建包含自增主键的表  
CREATE TABLE increnment_test (  id INT(11) NOT NULL AUTO_INCREMENT PRIMARY KEY,  col1 INT(11) NOT NULL,col2 INT(11) NOT NULL,col3 INT(11) NOT NULL,UNIQUE KEY (col1)
) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8mb4;

2.自增值存储机制

1.MyISAM 引擎的自增值保存在数据文件中。

2.Innodb 引擎

● 在 MySQL 5.7 及之前的版本,自增值保存在内存里。每次重启后,第一次打开表的时候,都会去找自增值的最大值 max(id),然后将 max(id) + 1 作为这个表当前的自增值。
● 在 MySQL 8.0 版本,将自增值的变更记录在了 redo log 中,重启的时候依靠 redo log 恢复重启之前的值。

-- 1.往increnment_test表中插入2条数据
INSERT INTO increnment_test (col1, col2, col3) VALUES    (1, 1, 1),    (2, 2, 2);-- 2.此时id值为2,AUTO_INCREMENT值为3。我们删除id为2的记录
delete from increnment_test where id = 2;-- 3.立马重启实例,重新插入记录。此时id为2。
INSERT INTO increnment_test (col1, col2, col3) VALUES    (2, 2, 2);-- 4.不重启实例,删除id为2的记录后,重新插入数据,id应为3。

3.自增值修改机制

在MySQL中,可以使用AUTO_INCREMENT关键字来指定ID字段为自增ID字段。当向表中插入一条记录时,MySQL将自动为该记录的ID字段生成一个新的自增ID值,并将该值保存到该记录的ID字段中。具体规则如下:

● 如果ID字段未指定具体的值,则将当前AUTO_INCREMENT值并将其填入自增字段,并生成新的自增值
● 如果ID字段已指定具体的值,则直接使用指定的值作为 ID 字段的值,而不会生成新的 AUTO_INCREMENT 值。

根据要插入的值和当前自增值的大小关系,自增值的变更结果也会有所不同。

● 如果插入值小于当前自增值,那么直接使用插入值填入ID字段,自增值不变;
● 如果插入值大于当前自增值,那么除了直接使用插入值填入ID字段外,自增值需修改为插入值+1;

上述”插入值+1‘不是直接使用”插入值“+1,是auto_increment_offset(自增初始值)以 auto_increment_increment(自增步长)为步长,持续累加,直到找到大于插入值的值,作为新的自增值。

4.自增值修改流程

上述我们了解了自增值的存储机制与修改机制,自增值修改是在哪个环境呢?那需要了解自增值修改流程。

INSERT INTO increnment_test (col1, col2, col3) VALUES (3, 3, 3);

以上述SQL为例,我们假如数据库里已经有2条数据了,它的执行流程如下:

● 执行器调用 InnoDB 引擎接口将分析器优化后的SQL传入,并将值(3,3,3)一起传过去。
● InnoDB 发现用户没有指定自增 id列,会先获取表increnment_test当前的自增值3;
● 将ID列补充完整,并且将自增值填入(3,3,3,3)
● 然后将表的自增值改成4;
● 继续执行插入数据操作;

自增字段值的生成是由存储引擎自动完成的,而不是由优化器完成的。因此,在执行 SQL 语句时,即使未指定自增字段列,也不会对性能产生任何影响。

5.导致自增值不连续的原因

5.1 唯一键冲突

比如increnment_test中已经存在了col1为3的记录,我们继续插入col1为3的记录,此时会出现唯一键冲突插入报错,但是没有将自增值再改回去。重新插入col1为4的值,此时对应的id为5;
如下操作流程:

5.1.1.检查数据

select * from increnment_test;

在这里插入图片描述

5.1.2.插入col1为3的数据

INSERT INTO increnment_test (col1, col2, col3) VALUES    (3, 3, 3);

5.1.3.插入col1为4的记录

INSERT INTO increnment_test (col1, col2, col3) VALUES    (4, 4, 4);
select * from increnment_test;

在这里插入图片描述

5.2.事务回滚

开启一个事务插入col1为6的数据,然后进行回滚。回滚后重新插入col1为6的记录,此时col1为6对应的id值为7。

BEGIN;    
INSERT INTO increnment_test (col1, col2, col3) VALUES      (6, 6, 6);   
ROLLBACK; BEGIN;    
INSERT INTO increnment_test (col1, col2, col3) VALUES      (6, 6, 6);   
COMMIT;

在这里插入图片描述

5.3.批量插入数据

对于批量插入数据的语句,MySQL有一个批量申请自增 id 的策略:

● SQL语句执行过程中,第1次申请自增 id,会分配 1 个;
● 1 个用完以后,第2次申请自增 id,会分配 2 个;
● 2 个用完以后,第3次申请自增 id,会分配 4 个;

依此类推,同一个语句去申请自增 id,每次申请到的自增id个数都是上一次的两倍。

drop table increnment_test2;
create table increnment_test2 like increnment_test;INSERT INTO increnment_test2 (col1, col2, col3) SELECTcol1, col2, col3 FROM increnment_test;INSERT INTO increnment_test2 (col1, col2, col3)
VALUES (8, 8, 8);  SELECT * FROM increnment_test2;

在这里插入图片描述

因为increnment_test2表中批量插入了5条数据,按照自增ID的批量申请策略,5条数据分3次进行申请:
第1次:id-1
第2次:id-2、3
第3次:id-4、5、6、7
由于只有5条记录,所以只使用了4、5被浪费了。当我们在次插入数据时,AUTO_INCREMENT从8开始。

这篇关于Mysql中自增主键是如何工作的的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/418838

相关文章

MySQL中读写分离方案对比分析与选型建议

《MySQL中读写分离方案对比分析与选型建议》MySQL读写分离是提升数据库可用性和性能的常见手段,本文将围绕现实生产环境中常见的几种读写分离模式进行系统对比,希望对大家有所帮助... 目录一、问题背景介绍二、多种解决方案对比2.1 原生mysql主从复制2.2 Proxy层中间件:ProxySQL2.3

MySQL 索引简介及常见的索引类型有哪些

《MySQL索引简介及常见的索引类型有哪些》MySQL索引是加速数据检索的特殊结构,用于存储列值与位置信息,常见的索引类型包括:主键索引、唯一索引、普通索引、复合索引、全文索引和空间索引等,本文介绍... 目录什么是 mysql 的索引?常见的索引类型有哪些?总结性回答详细解释1. MySQL 索引的概念2

setsid 命令工作原理和使用案例介绍

《setsid命令工作原理和使用案例介绍》setsid命令在Linux中创建独立会话,使进程脱离终端运行,适用于守护进程和后台任务,通过重定向输出和确保权限,可有效管理长时间运行的进程,本文给大家介... 目录setsid 命令介绍和使用案例基本介绍基本语法主要特点命令参数使用案例1. 在后台运行命令2.

MySQL中EXISTS与IN用法使用与对比分析

《MySQL中EXISTS与IN用法使用与对比分析》在MySQL中,EXISTS和IN都用于子查询中根据另一个查询的结果来过滤主查询的记录,本文将基于工作原理、效率和应用场景进行全面对比... 目录一、基本用法详解1. IN 运算符2. EXISTS 运算符二、EXISTS 与 IN 的选择策略三、性能对比

MySQL常用字符串函数示例和场景介绍

《MySQL常用字符串函数示例和场景介绍》MySQL提供了丰富的字符串函数帮助我们高效地对字符串进行处理、转换和分析,本文我将全面且深入地介绍MySQL常用的字符串函数,并结合具体示例和场景,帮你熟练... 目录一、字符串函数概述1.1 字符串函数的作用1.2 字符串函数分类二、字符串长度与统计函数2.1

SQL Server跟踪自动统计信息更新实战指南

《SQLServer跟踪自动统计信息更新实战指南》本文详解SQLServer自动统计信息更新的跟踪方法,推荐使用扩展事件实时捕获更新操作及详细信息,同时结合系统视图快速检查统计信息状态,重点强调修... 目录SQL Server 如何跟踪自动统计信息更新:深入解析与实战指南 核心跟踪方法1️⃣ 利用系统目录

MySQL 内存使用率常用分析语句

《MySQL内存使用率常用分析语句》用户整理了MySQL内存占用过高的分析方法,涵盖操作系统层确认及数据库层bufferpool、内存模块差值、线程状态、performance_schema性能数据... 目录一、 OS层二、 DB层1. 全局情况2. 内存占js用详情最近连续遇到mysql内存占用过高导致

Mysql中设计数据表的过程解析

《Mysql中设计数据表的过程解析》数据库约束通过NOTNULL、UNIQUE、DEFAULT、主键和外键等规则保障数据完整性,自动校验数据,减少人工错误,提升数据一致性和业务逻辑严谨性,本文介绍My... 目录1.引言2.NOT NULL——制定某列不可以存储NULL值2.UNIQUE——保证某一列的每一

解密SQL查询语句执行的过程

《解密SQL查询语句执行的过程》文章讲解了SQL语句的执行流程,涵盖解析、优化、执行三个核心阶段,并介绍执行计划查看方法EXPLAIN,同时提出性能优化技巧如合理使用索引、避免SELECT*、JOIN... 目录1. SQL语句的基本结构2. SQL语句的执行过程3. SQL语句的执行计划4. 常见的性能优

SQL Server 中的 WITH (NOLOCK) 示例详解

《SQLServer中的WITH(NOLOCK)示例详解》SQLServer中的WITH(NOLOCK)是一种表提示,等同于READUNCOMMITTED隔离级别,允许查询在不获取共享锁的情... 目录SQL Server 中的 WITH (NOLOCK) 详解一、WITH (NOLOCK) 的本质二、工作