MySQL中的主键自增机制详情
作者:昨天的风 发布时间:2024-01-18 23:50:54
主键自增
MySQL 提供了主键自增机制 AUTO_INCREMENT
. 对主键使用, 保证了主键的唯一性.
注意:自增长必须与主键字段配合使用
。
默认的主键的起始值为 1, 每次增量为 1
, 也可以手动指定
其自增起始值 auto_increment_offset 和自增步长 auto_increment_increment.
-- 设置主键自增
CREATE TABLE USER(
id INT UNSIGNED AUTO_INCREMENT,
...
)
-- 在创建表时指定自增起始值
CREATE TABLE xx{
}ENGINE=INNODB auto_increment=100 DEFAULT CHARSET=utf8;
-- 修改自增起始值
alter table people AUTO_INCREMENT = 20;
-- 基于会话级别:
show session variables like 'auto_inc%';-- 查看步长
set session auto_increment_increment=2; -- 修改自增步长
-- 基于全局级别:(会影响下一次登录的值)
show global variables like 'auto_inc%';-- 查看步长
set global auto_increment_increment=5;-- 修改自增步长
注意:
在 InnoDB 存储引擎中, 自增长值的列必须是索引, 同时必须是索引的第 1 个列. 如果不是第 1 个列, 则 MySQL 数据库会抛出异常. 对于 MyISAM 无此要求。
自增长每次递增 1, 说明是数值型, 可以是整数, 也可以是浮点数.
由于种种原因, 自增值可以保证增长趋势, 但并不能保证连续.
不会影响自增长的命令:DELETE FROM xxx;
可以影响自增长的命令:TRUNCATE TABLE xxx;
使用起来倒是很简单,但是对于主键自增机制的这些问题,你了解吗?
自增主键保存在哪里?
自增主键如何实现自增的?
自增主键是什么时候自增的?
自增主键一定是连续自增的吗?
自增主键可以人为修改吗?
自增主键保存在哪里
首先需要知道的是,自增主键机制是存储引擎实现的,所以不同的存储引擎对于自增值的保存策略不同.
MyISAM 的自增值保存在数据文件中.
InnoDB 的自增值,
保存在内存里
, 一直到了 MySQL 8.0 后, 才有了自增值的持久化
的能力, 也就是才保存到文件中,实现了如果发生重启, 表的自增值可以恢复为 MySQL 重启前的值.
具体是:在 MySQL <= 5.7
时, 自增值保存在内存里, 没有持久化. 当 MySQL 重启后, 第一次打开某个数据表的时候, 都会去找该表中主键字段的自增值的最大值 max(id), 然后将 max(id)+1
作为这个表当前的自增值.
但是这样就会存在一个问题,比如 : 如果一个表的 id 最大是 10, 此时的 AUTO_INCREMENT=11. 当删除 id=10 的记录时, 此时 AUTO_INCREMENT 还是 11. 但如果马上重启 MySQL, 重启后这个表的 AUTO_INCREMENT 就变为 10 了. ( maxid = 9, 9+1=10 ) 即 MySQL 重启可能会修改表的 AUTO_INCREMENT 的值.
在 MySQL 8.0
版本, 将自增值的变更记录在了 redo log
中, 重启的时候依靠 redo log 恢复重启之前的值. 所以不会出现上述问题.
自增值修改机制
在 MySQL 中, 如果字段 id 被定义为 AUTO_INCREMENT, 在插入一行数据的时候, 自增值的操作如下:
如果插入数据时, id 字段指定为
0, null 或未指定值
, 那么就把这个表当前的 AUTO_INCREMENT 值给自增字段, (没给值, 则使用自增长值
)如果插入数据时, id 字段指定了具体的值, 就直接使用 SQL 语句里指定的值. (
给定了值, 就使用给定值
)根据要插入的值和当前自增值的大小关系, 自增值的变更结果也会有所不同.
假设, 要插入的值是 X, 当前的自增值是 Y.
如果 X < Y
, 那么这个表的自增值不变
.
如果 X ≥ Y
, 就需要把当前表的自增值修改为新的自增值. 新的自增值生成方式是:从 auto_increment_offset 开始, 以 auto_increment_increment 为步长, 持续叠加, 直到找到第一个大于 X 的值, 作为新的自增值.
也就是,这种情况下步长也参与了影响。
自增值的修改时机
假设, 有一个表 t , 有 A, B, C 三个字段, 字段 A 是主键, 且自增, 字段 C 有唯一约束.
当前表 t 中有一条记录为 (1,1,1) , 此时执行一条插入语句 insert into t values(null, 1, 1);
那么这个语句的执行流程就是:
执行器调用 InnoDB 引擎的接口,写入一行, 传入的这一行的值是 (null,1,1);
InnoDB 发现用户没有指定自增 id 的值, 所以会获取表 t 当前的自增值 2,
然后 InnoDB 会将传入的行的值改成 (2,1,1);
然后将表的自增值改成 3,
然后执行插入数据操作, 由于 C 字段已经存在 = 1 的记录, 所以报 Duplicate key error, 并返回.
最后的结果可以看到, 这个表的自增值改成 3, 是在真正执行插入数据的操作之前进行的
.
这个语句真正执行的时候, 因为碰到唯一键 C 冲突, 所以 id=2 这一行并没有插入成功, 但也没有将自增值再改回去. 所以, 在这之后, 再插入新的数据行时, 拿到的自增 id 就是 3. 也就是说, 出现了自增主键不连续的情况.
总结:自增主键不连续
有如下两种情况 :
唯一约束冲突
导致自增主键不连续.事务回滚
也会导致自增主键不连续.
如何修改自增主键值
有如下几种修改方式:
使用
alter table 表名 AUTO_INCREMENT = ?
来修改自增值的起始值。在创建表时设置 AUTO_INCREMENT=? 自增值的起始值。
来源:https://juejin.cn/post/7135460563919831053


猜你喜欢
- SQL执行步骤的具体分析先来看执行语句的顺序(8)select (9)distinct A (1)from Ta (3)join Tb (2
- 一、 了解postman1. 什么是postman?------ 软件测试用来做接口测试的工具。2. 如何下载postman--
- 核心代码function convert2utf8($string) { return iconv(&
- 在布尔索引中,我们将根据 DataFrame 中数据的实际值而不是它们的行/列标签或整数位置来选择数据子集。在布尔索引中,我们使用布尔向量来
- 有时候我们可能需要import另一个路径下的python文件,例如下面这个目录结构,我们想要在_train.py里import在networ
- 前言最小二乘 * east Square Method,做为分类回归算法的基础,有着悠久的历史(由马里·勒让德于1806年提出)。它通过最小化
- 需求分析根据原始数据,计算出累计和、回撤、连续正确、连续错误、连续正确值与连续错误值6项数据,其中原始数据大于等于0认定为正确,原始数据小于
- 尽管现代浏览器已经支持了众多的CSS3属性,但是大部分设计师和开发人员貌似依然在关注于一些很“主流”的属性,如border-radius、b
- 引子平常的文本处理工作中,我经常会遇到这么一种情况:用python判断一个string是否包含一个list里的元素。这时候使用python的
- 本文实例为大家分享了python读取视频流提取视频帧的具体代码,供大家参考,具体内容如下方法一:通过imageio库和skimage库1.
- 二分查找算法:简单的说,就是将一个数组先排序好,比如按照从小到大的顺序排列好,当给定一个数据,比如target,查找target在数组中的位
- 前言为了往我们写好的Python代码传入参数,有很多种方法,比如使用input获取从DOS 输入的参数,又或者读取txt 文件中的字符作为参
- 带你走进数据类型一:整数、浮点数Python中整数和浮点数的定义以及运算和C++都是一样的,我在这里就不需多说了,我就说明一点:Python
- 蚁群算法简介蚁群算法(Ant Clony Optimization, ACO)是一种群智能算法,它是由一群无智能或有轻微智能的个体(Agen
- 使用folium excel 绘制point制作内容根据气象台资料获得的点进行绘制对一个特殊的点做特别的标注数据来源#!/usr/bin/e
- 关于SQL查询效率,100w数据,查询只要1秒,与您分享:机器情况:p4: 2.4内存: 1 Gos: windows 2003数据库:SQ
- 概述何为路由守卫?路由守卫有点类似于ajax的请求 * ,就是请求发送之前先给你拦截住做一些事情之后再去发送请求,同样这里的路由守卫意思差不
- 一、SQLite简介SQLite是一个包含在C库中的轻量级数据库。它并不需要独立的维护进程,并且允许使用非标准变体(nonstandard
- Tensorflow数据读取有三种方式:Preloaded data: 预加载数据Feeding: Python产生数据,再把数据喂给后端。
- 跟着节奏继续来探索fixtures的灵活性。一、一个测试函数/fixture一次请求多个fixture在测试函数和fixture函数中,每一