面试官问:MySQL 的自增 ID 用完了,怎么办?

★★★建议星标我们★★★
Java进阶架构师★“星标”!这样才不会错过每日进阶架构文章呀 。
面试官问:MySQL 的自增 ID 用完了,怎么办?文章插图
面试官问:MySQL 的自增 ID 用完了,怎么办?文章插图
2020年Java原创面试题库连载中
【000期】Java最全面试题库思维导图
【020期】JavaSE系列面试题汇总(共18篇)
【028期】JavaWeb系列面试题汇总(共10篇)
【042期】JavaEE系列面试题汇总(共13篇)
【049期】数据库系列面试题汇总(共6篇)
【053期】中间件系列面试题汇总(共3篇)
【065期】数据结构与算法面试题汇总(共11篇)
【076期】分布式面试题汇总(共10篇)
【077期】综合面试题系列(一)
【078期】综合面试题系列(二)
【079期】综合面试题系列(三)
【080期】综合面试题系列(四)
【081期】综合面试题系列(五)
【082期】综合面试题系列(六)
【083期】综合面试题系列(七)
【084期】综合面试题系列(八)
【085期】综合面试题系列(九)
【086期】综合面试题系列(十)
【087期】综合面试题系列(十一)
【088期】综合面试题系列(十二)
【089期】综合面试题系列(十三)
更多内容 , 点击上面蓝字查看
面试官问:MySQL 的自增 ID 用完了,怎么办?文章插图
面试官问:MySQL 的自增 ID 用完了,怎么办?文章插图
既然这块知识点不清楚 , 那回头就自己动手实践下 。
首先 , 创建一个最简单的表 , 只包含一个自增 id , 并插入一条数据 。
create table t0(id int unsigned auto\_increment primary key) ;insert into t0 values;通过 show 命令 show create table t0;查看表情况
CREATE TABLE \`t0\` ( \`id\` int(10) unsigned NOT AUTO\_INCREMENT, PRIMARY KEY (\`id\`)) ENGINE=InnoDB AUTO\_INCREMENT=2 DEFAULT CHARSET=utf8可以发现 AUTO_INCREMENT 已经自动变成 2 , 这离用完还有很远 , 我们可以算下最大当前声明的自增 ID 最大是多少 , 由于这里定义的是 intunsigned , 所以最大可以达到 2 的 32 幂次方 - 1 = 4294967295
这里有个小技巧 , 可以在创建表的时候 , 直接声明 AUTO_INCREMENT 的初始值
create table t1(id int unsigned auto\_increment primary key) auto\_increment = 4294967295;insert into t1 values;同样 , 通过 show 命令 , 查看 t1 的表结构
CREATE TABLE \`t1\` ( \`id\` int(10) unsigned NOT AUTO\_INCREMENT, PRIMARY KEY (\`id\`)) ENGINE=InnoDB AUTO\_INCREMENT=4294967295 DEFAULT CHARSET=utf8可以发现 , AUTO_INCREMENT 已经变成 4294967295 了 , 当想再尝试插入一条数据时 , 得到了下面的异常结果
17:28:03 insert into t1 values Error Code: 1062. Duplicate entry '4294967295' for key 'PRIMARY' 0.00054 sec说明 , 当再次插入时 , 使用的自增 ID 还是 4294967295 , 报主键冲突的错误 。
4294967295 , 这个数字已经可以应付大部分的场景了 , 如果你的服务会经常性的插入和删除数据的话 , 还是存在用完的风险 , 建议采用 bigint unsigned , 这个数字就大了 。
不过 , 还存在另一种情况 , 如果在创建表没有显示申明主键 , 会怎么办?
如果是这种情况 , InnoDB 会自动帮你创建一个不可见的、长度为 6 字节的 row_id , 而且 InnoDB 维护了一个全局的 dictsys.row_id , 所以未定义主键的表都共享该 row_id , 每次插入一条数据 , 都把全局 row_id 当成主键 id , 然后全局 row_id 加 1