news 2026/4/15 8:09:38

MySQL自增id超过int最大值的场景

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
MySQL自增id超过int最大值的场景

点击标题下「蓝色微信名」可快速关注

数据库的主键我们有时候会用自增列,但是自增都会有个上限,如果达到怎么办?技术社群的这篇文章《MySQL自增id超过int最大值怎么办?》就给我们讲解了MySQL数据库自增列达到上限该怎么办,借鉴学习下。

一、故事背景

今天运维反馈有一个设备在后台查不到,我第一时间怀疑可能是数据出了问题,导致服务报错了没有入库。我拿着日志去本地请求接口,发现程序是没有报错的,我们的逻辑是先将唯一id放到redis里面,如果redis没有值就insert,有就update,做了一层缓存,估计是这样的话批量插入和更新数据库会快一点。然后我看redis是有值的,以为是redis和数据库数据不一致问题,我就把redis的key删了,重新再跑一下,结果打印了insert语句,但是没有插入到数据,看来事情并没有那么简单。

二、问题分析

因为数据表很大,有5E+数据,我第一反应是mysql表数据量可能爆了,但是查了下好像没有太大限制,

再认真看了下表的自增id,这个数字让人有点熟悉的,这个不就是int的最大值吗。意思是因为自增id超过了int,所以插入失败了,id设的就是int类型,还有个小彩蛋,目前数据库设的int长度是50,但是根本没什么鸟用。

知道了问题在哪,但是这个问题处理起来很麻烦,因为数据量太大了,先请教一下deepseek。

三、方案处理

deepseek给我提供了三个方案,第一个是最简单粗暴的改BIGINT,不用迁移数据,但是会全程锁表。第二个分布式ID需要重新设计表,需要把数据迁移到新表,而且还要redis等支撑。第三个分库分表就更麻烦了,分库分表需要引入框架,不按照分片查询还需要引入ES,引入了ES还需要引入同步mysql和ES的中间件logstash等。

但是改bigint估计锁表太久,我先看看有没有其他办法先紧急处理下数据。但是按理说int最大值是21E+,数据表数据才5E+,按理说是用不完的。结果我看到自增的id值居然是不连续的。

按理说自增id应该是一个接着一个,不会有空隙的,后面查了一下由于数据库自增id有个高性能策略,设置了id就不一定连续。

后面又查了下有没有一键把数据表id重排的方法,结果也是没有的。最后我是写了一个存储过程先将最后100万的id清理出来,可以先顶个几天,后面再想办法处理,

BEGIN DECLARE start_id INT DEFAULT 1; DECLARE end_id INT DEFAULT 100000; DECLARE current_batch INT DEFAULT 0; WHILE start_id <= end_id DO -- 更新临时表中的ID UPDATE table SET id = start_id + 1 WHERE id = (select original_id from ( SELECT id AS original_id FROM table ORDER BY id DESC LIMIT 1) as test); SET start_id = start_id + 1; END WHILE;END

最后重新设置自增值,如果自增值已经存在,则会跳到max(id)+1,

-- 重置自增值ALTER TABLE your_table AUTO_INCREMENT = max(id)+1;

清理了大概500万的id段出来,然后我怀疑id间隔这么大是因为并发太高导致的。一开始程序是单线程,消费到500条就批量入库,但是后面发现单线程消费比较慢,数据量太多消费有点延迟。后面改成java批量消费,配置了30个消费者。接着我尝试了一下减少消费者数量,设置成15个,id的间隔真的变小了。

四、设置BIGINT

节后回来发现id还剩200万,讨论到最后还是把id的数据类型从int改成bigint,

ALTER TABLE xxx MODIFY id BIGINT UNSIGNED NOT NULL AUTO_INCREMENT

UNSIGNED无符号位,不算负数,可以增加一倍数据,NOT NULL非空 AUTO_INCREMENT自增。

在测试环境有一亿数据,修改id的类型大概用了一个小时,现网我估计也是用6-7个小时也差不多了。结果改了一晚上都还没改好,然后我找了一个可以查询sql进度的语句,

SELECT EVENT_NAME, WORK_COMPLETED, WORK_ESTIMATED, ROUND(WORK_COMPLETED/WORK_ESTIMATED*100, 2) AS "Progress (%)" FROM performance_schema.events_stages_current;

跑了十几个小时居然还不到50%,而且还越跑越慢。对比了一下测试环境和现网环境的buffer_pool等数据也是设置正常。

估计是索引树变大插入的数据要花多不少时间,还有一个就是现网数据库还有其他线程会抢占CPU导致速度缓慢。

统计了一下后面的数据大概是1个小时完成1.5%左右,

周一晚上执行的,但到周四早上上班的时候才跑完,用了2天多一点。

五、总结

之前刷到一篇文章《字节面试:MySQL自增ID用完会怎样?》,评论区都说有没有用完的,结果我真用完了,就感觉有点不可思议。总结一下有几个原因:

1、数据量确实很大,有5E多数据,然后并发也很高。其实当初他们设计的时候也预料过这个问题,所以设了个int长度50,但是这个长度没起作用- -所以设计数据库的时候一定要做好,不然几亿数据改个字段类型要2天。

2、数据库的自增id策略选了高性能策略,导致并发高的时候id间隔很大。30个消费者异步处理,10条数据大概用了100个id的间隔,消耗太快了。所以这里存在一个时间和空间的取舍,使用多线程还是挺危险的操作,要谨慎一点。

还有一个小插曲,因为系统两天没消费数据,kafka的数据堆积了很多,然后我将消费者数量从30个改成50个,跑了两天,kafka还是有1天的延迟,看来麻木添加消费者数量已经没啥提升的作用了,想起八股文说多线程弄太多反而增加上下文切换的时间浪费,跟这个同理。

通过改造成sql批量消费,消费速度马上提上去了。程序的消费策略:单线程批量500个开始消费 ——> 30个线程单个消费 ——> 30个线程批量50个开始消费。

所以说多线程异步+批量操作的策略还是很重要的!不过多线程一定要注意异步问题。

如果您认为这篇文章有些帮助,还请不吝点下文章末尾的"点赞"和"在看",或者直接转发朋友圈,

可以到各大平台找我,

  • 微信公众号:@bisal的个人杂货铺

  • 腾讯云开发者社区:@bisal的个人杂货铺

  • 头条号:@bisal的个人杂货铺

  • CSDN:@bisal

  • ITPub:@bisal

  • 墨天轮:@bisal

  • 51CTO:@bisal

  • 小红书:@bisal

  • 抖音:@bisal

近期更新的文章:

《Instagram十亿级"用户名已被占用"背后的架构设计》

《为轮子造轮子的教训经验场景》

《国子监辟雍内部匾额少了一块么?》

《英超第二十一轮》

《冬季奥运会观赛赛程》

近期Vlog:

《千岛湖》

《Skyline Luge》

《新疆之行(红山体育馆 - 国际大巴扎 - 红山公园 - 天山天池)》

《新疆之行(天马浴河 - 哈因塞 - 那拉提 - 依提根塞)》

《新疆之行(六星街 - 伊昭公路 - 夏塔)》

热文鉴赏:

《揭开"仿宋"和"仿宋_GB2312"的神秘面纱》

《Linux的"aarch"是多了个"a"?》

《中国队“自己的”世界杯》

《你不知道的C罗-Siu庆祝动作》

《大阪环球影城避坑指南和功略》

《推荐一篇Oracle RAC Cache Fusion的经典论文》

《"红警"游戏开源代码带给我们的震撼》

文章分类和索引:

《公众号1900篇文章分类和索引》

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/3 6:39:06

Orin十年演进

下面这份内容&#xff0c;不是 NVIDIA 产品路线图复述&#xff0c;也不是算力参数对比&#xff0c;而是站在Orin 作为“一代自动驾驶/机器人计算平台的代表性节点”的高度&#xff0c;对它在未来十年中的角色演进与历史定位做一次结构性判断。&#x1f9e0;⚙️ Orin 十年演进&…

作者头像 李华
网站建设 2026/4/10 5:15:32

车联网十年演进

下面这份内容&#xff0c;不是“V2X 技术白皮书”&#xff0c;也不是通信协议时间线&#xff0c;而是站在**“车联网作为智能交通与自动驾驶的社会级基础设施”视角&#xff0c;对未来十年的一次结构性演进判断**。&#x1f310;&#x1f697; 车联网十年演进&#xff08;2025–…

作者头像 李华
网站建设 2026/4/13 7:37:20

车规级芯片十年演进

下面这份内容&#xff0c;不是芯片参数对比&#xff0c;也不是厂商 Roadmap&#xff0c;而是站在 “车规级芯片作为智能汽车系统底座”的高度&#xff0c;对未来十年的一次结构性演进判断。&#x1f9e0;&#x1f697; 车规级芯片十年演进&#xff08;2025–2035&#xff09; 一…

作者头像 李华
网站建设 2026/4/14 9:26:50

LLVM后端入门9:JIT支持

目标机器的实现可以选择性地包含一个即时(JIT)代码生成器,该生成器会将机器码和辅助结构作为二进制输出进行生成,这些输出可直接写入内存。要实现这一点,可通过执行以下步骤来实现 JIT 代码生成: 编写 XXXCodeEmitter.cpp 文件,其中包含一个机器函数 pass,用于将目标机…

作者头像 李华
网站建设 2026/4/11 22:22:51

系统软件缺少comct332.ocx无法启动 免费下载方法分享

在使用电脑系统时经常会出现丢失找不到某些文件的情况&#xff0c;由于很多常用软件都是采用 Microsoft Visual Studio 编写的&#xff0c;所以这类软件的运行需要依赖微软Visual C运行库&#xff0c;比如像 QQ、迅雷、Adobe 软件等等&#xff0c;如果没有安装VC运行库或者安装…

作者头像 李华
网站建设 2026/4/14 23:08:00

你不是在和AI竞争,你是在和“不用AI的自己”

导言&#xff1a;被误读的竞争关系 凌晨三点的告警邮件、重复的冒烟测试、永无止境的回归用例...当ChatGPT写出第一条测试脚本时&#xff0c;测试圈掀起海啸式恐慌。但真正需要警惕的并非AI&#xff0c;而是我们面对技术变革时固化的思维模式——软件测试的竞争本质&#xff0…

作者头像 李华