首页
学习
活动
专区
工具
TVP
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往

MySQL主键为什么不连续

主键可以主键索引尽量的保持递增顺序插入,避免页分裂,索引更加紧凑。 主键保存在何处?...不同的引擎对于增值的保存策略不同: MyISAM引擎的增值保存在数据文件中 InnoDB引擎的增值保存在内存里,但是在MySQL8.0以后,该自增值才可以被持久化:MySQL5.7以前,增值没有持久化每次重启后第一次打开表的时候...(默认值是1)开始,以auto_increment_increment(默认值是1)为步长,持续叠加,直到找到第一个大于X的值,作为新的增值 唯一键冲突导致主键不连续 insert into t...事务回滚导致主键不连续 set autocommit=0; begin; insert into t values(null, 2, 2); rollback; show create table...) 在innodb_autoinc_lock_mode参数为1的情况下,如果大批量插入数据也会造成id的不连续

8.4K20

MySQL主键一定是连续

全文摘要 结合实例分析了增值保存在哪里,增值的修改策略,以及增值不连续的四个场景,希望对各位小伙伴们有所帮助~ 众所周知,主键可以聚集索引尽量地保持递增顺序插入,避免了随机查询,从而提高了查询效率...但实际上,MySQL主键并不能保证一定是连续递增的。...非也~ 了解过分布式 id 的小伙伴一定知道,为了避免两个库生成的主键发生冲突,我们可以一个库的 id 都是奇数,另一个库的 id 都是偶数 这个奇数偶数其实是通过 auto_increment_offset...回退回去的话不就不会发生 id 不连续了吗? 事实上,这么做的主要原因是为了提高性能。 我们直接用反证法来验证:假设 MySQL 在事务回滚的时候会把增值改回去,会发生什么?...增值不连续场景 4 对于批量插入数据的语句,MySQL 有一个批量申请 id 的策略: 语句执行过程中,第一次申请 id,会分配 1 个; 1 个用完以后,这个语句第二次申请 id,会分配

3.8K10
  • 您找到你想要的搜索结果了吗?
    是的
    没有找到

    39 | 主键不连续

    主键可以主键索引尽量地保持递增顺序插入,避免了页分裂,因此索引更紧凑。...比如,双 M 的主备结构里要求双写的时候,我们就可能会设置成 auto_increment_increment=2,一个库的 id 都是奇数,另一个库的 id 都是偶数,避免两个库生成的主键发生冲突...增值的不连续情况 1....因此,InnoDB 放弃了这个设计,语句执行失败也不回退 id。也正是因为这样,所以才只保证了 id 是递增的,但不保证是连续的。...锁的优化 id 锁并不是一个事务锁,而是每次申请完就马上释放,以便允许别的事务再申请 在 MySQL 5.0 版本的时候,锁的范围是语句级别。

    4.8K10

    mysql 主键语句_MySQL 主键

    连续性 插入成功时,其数据的 ID 和前一次插入成功时数据的 ID 相邻。 主键的单调性 为何会有单调性的问题? 这主要跟主键最大值的获取方式,以及存放位置有关系。...主键最大值怎么取的?存放到哪里?...从 MySQL 8.0 开始,主键最大值会在每次修改后写入到 redo log,并且在每个检查点写入引擎私有的系统表。 如果是正常重启,则读取系统表里的值。...主键插入时的连续性 这里不考虑由于删除导致的连续性问题 为何会有连续性问题? 这主要是跟插入事务回滚有关系。 对于两个插入事务,事务 A 先执行插入语句,之后事务 B 执行插入语句。...参考文档 为什么 MySQL主键不单调也不连续 https://database.51cto.com/art/202004/614923.htm 《MySQL技术内幕——InnoDB存储引擎》

    10.8K10

    MySQL重大Bug!主键竟然不是连续递增

    很多低级开发工程师都想当然觉得主键是严格连续递增的,但事实真的如此吗?...比如,双M架构要求双写时,可能设置成auto_increment_increment=2,一个库的id都是奇数,另一个库的id都是偶数,避免两个库生成的主键发生冲突。...此后再成功插入新数据,拿到id就是3了 如你所见,主键不连续了!所以唯一键冲突是导致主键id不连续的一大原因。 事务回滚是二大原因。...要解决该问题,有如下方案: 原库的批量插入数据语句,固定生成连续id值 所以,锁直到语句执行结束才释放,就是为了达此目的 在binlog里把插入数据的操作都如实记录进来,到备库执行时,不依赖主键去生成...这是主键id不连续的三大原因。

    2.6K00

    MySQL重大Bug!主键竟然不是连续递增

    很多低级开发工程师都想当然觉得主键是严格连续递增的,但事实真的如此吗?...比如,双M架构要求双写时,可能设置成auto_increment_increment=2,一个库的id都是奇数,另一个库的id都是偶数,避免两个库生成的主键发生冲突。...此后再成功插入新数据,拿到id就是3了 如你所见,主键不连续了!所以唯一键冲突是导致主键id不连续的一大原因。 事务回滚是二大原因。...要解决该问题,有如下方案: 原库的批量插入数据语句,固定生成连续id值 所以,锁直到语句执行结束才释放,就是为了达此目的 在binlog里把插入数据的操作都如实记录进来,到备库执行时,不依赖主键去生成...这是主键id不连续的三大原因。

    3.5K10

    mysql主键策略_MySQL 主键机制

    主键:特指在增列上定义的主键。 主键的优点是主键索引保持递增顺序的插入,避免页分裂,索引更加紧凑。 1. 增值保存在哪? 不同的存储引擎保存增值的策略不一样; a....对于MyISAM引擎,增值保存在数据文件中; b. Innodb引擎,mysql5.7之前,增值保存在内存中,而且不会持久化增值。...每次重启后第一次打开表,都会去查找增值的最大值max(id), 并设置表当前增值为max(id) + 1; mysql8.0, 增值变更记录在了redo log中,重启时依靠redo log恢复重启之前的值...增值修改发生在插入数据的操作之前,如果插入失败,增值不会再修改回去; b. 事务回滚也不会将自增值修改回去; c. 为了减少id锁带来的性能影响,mysql不会修改回去之前的增值; 4....而对于批量插入数据的语句(select … insert,replace … select 和 load data 语句),MySQL 有一个批量申请 id 的策略(注:该策略是导致 id 不连续的第三种原因

    9.4K50

    不懂就问:MySQL 主键一定是连续的吗?

    测试环境: MySQL版本:8.0 数据库表:T (主键id,唯一索引c,普通字段d) 如果你的业务设计依赖于主键的连续性,这个设计假设主键是连续的。...但实际上,这样的假设是错的,因为主键不能保证连续递增。 一、增值的属性特征: 1. 主键值是存储在哪的?...大家可以发现这个操作并没有进行原子操作,如果SQL语句执行失败,那么是不是就不会连续了呢?...所以,在这之后,再插入新的数据行时,拿到的 id 就是 3。也就是说,出现了主键不连续的情况。...五、主键值不连续情况:(批量插入) 批量插入数据的语句,MySQL 有一个批量申请 id 的策略: 语句执行过程中,第一次申请 id,会分配 1 个; 1 个用完以后,这个语句第二次申请

    17310

    主键不连续的几种情况

    //主键不连续的几种情况// 最近在极客时间上学习丁奇大佬的《MySQL 45讲》,这里结合自己的理解分享出来,喜欢的同学可以购买原版课程进行学习,里面的内容很丰富。...00 主键介绍 MySQL中的主键想必大家都不陌生,它是通过两个参数来控制的,分别是auto_increment_offset和auto_increment_increment,其中offset...02 事务回滚导致的键不连续 当我们使用回滚事务的时候,如果该事务内部使用了增值,那么同样会导致表主键出问题,示例如下: mysql> insert into t values (null...03 MySQL锁优化带来的不连续MySQL5.7中,参数innodb_autoinc_lock_mode被用来控制锁的模式,该参数可以设置为三个值:0、1、2. a、当该值为0的时候,...为了避免id不连续而造成的主从数据不一致,线上环境,建议设置成innodb_autoinc_lock_mode=2 ,并且 binlog_format=row.这样做,既能提升并发性,又不会出现数据一致性问题

    3.4K30

    线上MySQLid用尽怎么办?

    作者:废柴程序员 链接:https://www.jianshu.com/p/a6bc14005b52 MySQLid都定义了初始值,然后不断加步长。...那id用完,会怎么样? 图片 表定义增值id 表定义的增值达到上限后的逻辑是:再申请下一个id时,得到的值保持不变。...验证该结论:通过gdb修改系统的row_id。用gdb是为了便于复现问题,只能在测试环境使用。...所以应该在InnoDB表中主动创建主键:当表id到达上限后,再插入数据时会报主键冲突错误。 毕竟覆盖数据,就意味着数据丢失,影响数据可靠性;报主键冲突,插入失败,影响可用性。...因为MySQL使用了一个唯一数组 图片 给新线程分配thread_id时的逻辑: 图片 总结 每种id有各自的应用场景,在达到上限后的表现也不同: 表的id达到上限后,再申请时它的值就不会改变

    3.9K20

    线上MySQLid用尽怎么办?

    MySQLid都定义了初始值,然后不断加步长。虽然自然数没有上限,但定义了表示这个数的字节长度,计算机存储就有上限。...那id用完,会怎么样? 表定义增值id 表定义的增值达到上限后的逻辑是:再申请下一个id时,得到的值保持不变。...验证该结论:通过gdb修改系统的row_id。用gdb是为了便于复现问题,只能在测试环境使用。...所以应该在InnoDB表中主动创建主键:当表id到达上限后,再插入数据时会报主键冲突错误。 毕竟覆盖数据,就意味着数据丢失,影响数据可靠性;报主键冲突,插入失败,影响可用性。...因为MySQL使用了一个唯一数组 给新线程分配thread_id时的逻辑: 总结 每种id有各自的应用场景,在达到上限后的表现也不同: 表的id达到上限后,再申请时它的值就不会改变

    2.1K20

    线上MySQLid用尽怎么办?

    MySQLid都定义了初始值,然后不断加步长。虽然自然数没有上限,但定义了表示这个数的字节长度,计算机存储就有上限。...那id用完,会怎么样? 表定义增值id 表定义的增值达到上限后的逻辑是:再申请下一个id时,得到的值保持不变。...验证该结论:通过gdb修改系统的row_id。用gdb是为了便于复现问题,只能在测试环境使用。...所以应该在InnoDB表中主动创建主键:当表id到达上限后,再插入数据时会报主键冲突错误。 毕竟覆盖数据,就意味着数据丢失,影响数据可靠性;报主键冲突,插入失败,影响可用性。...因为MySQL使用了一个唯一数组 给新线程分配thread_id时的逻辑: 总结 每种id有各自的应用场景,在达到上限后的表现也不同: 表的id达到上限后,再申请时它的值就不会改变,进而导致继续插入数据时报主键冲突错误

    3.2K10

    深入剖析 MySQL

    怎么说呢,说他对,但是他也不完全对。 行为与限制 其实上面说的那种阻塞情况只是锁行为的其中一种,可以理解为锁就是一个接口,其具体的实现有多种。...连续模式 连续模式(Consecutive)是 MySQL 8.0 之前默认的模式,之所以提出这种模式,是因为传统模式存在影响性能的弊端,所以才有了连续模式。...基于 MySQL 默认 Binlog 格式从 Statement 到 Row 的变更,InnoDB 也将其锁的默认实现从连续模式,更换到了效率更高的交叉模式。...鱼和熊掌 但是如果你的 MySQL 版本仍然默认使用连续模式,但同时又想要提高性能,该怎么办呢?这个其实得做一些取舍。...如果你可以断定你的系统后续不会使用 Binlog,那么你可以选择将自锁的锁模式从连续模式改为交叉模式,这样可以提高 MySQL 的并发。

    3K20

    MySQL锁的探究

    2.5 MySQL锁是指在使用主键(Auto Increment)时,为了保证唯一性和正确性,系统会对字段进行加锁。这样可以确保同时插入多条记录时,每条记录都能够获得唯一的增值。...1)插入原理MySQL锁的实现机制是使用了一个名为"auto-increment lock"的互斥锁。...当使用INSERT语句插入一条新记录时,MySQL会自动为字段加锁,防止其他并发的插入操作同时获取相同的增值。这个锁是在内部实现的,不需要用户手动创建或管理。...能够产生一个轻量级的页面锁来保证insert的连续插入;对于Bulk Inserts无法确定插入的行数时采用表级别锁来保证insert的连续插入;2:interleaved(交叉模式):不采用表锁,...获取到auto_increment值时就会立即释放,跟事务无关;2)consecutive(连续模式)在连续模式下,InnoDB会根据当前执行的insert语句来判断是否使用表级别锁。

    10421
    领券