mysql删除数据后主键值不连续

mysql删除数据后主键值不连续,第1张

ID 这字段是表中的 主键吧?

那么 可能和其他表 有关联 如果改动了 就要出大问题了

当然了 要是单表的话你可以整

Delid = 26

delete from tablename where id = Delid

update tablename set id = id-1 where id >Delid

更新自增的主键值 在 mysql 行的通 其他的 数据库 没试过

-----------------------------------------------------

alter table tablename AUTO_INCREMENT = AUTO_INCREMENT - 1

将表的 自增 ID 值减 1 即可

你好哦..看到你的问题,我有如下见解:

你是想得到不连续行,如果要查询这个的话,有两种方式:

可以用到主键,要知道主键是唯一的而且也是连续的,所以你可以使用sql语句

select * from table(表名) where primarykey(主键) % 2 = 0

第二种可以利用行 页进行查询 (套用for循坏)

for(int page = 0 i<pageSum i++ ){

select * from table(表名)  limit page-1,1

}

当然还有很多查询方法,比如你在表中设置特定的字段进行查询,最后希望能采纳 谢谢哇!

第 4 篇 提过自增主键,让主键索引尽量 递增顺序插入 , 避免页分裂 ,索引 更紧凑 。

不能保证连续递增 。 什么情况出现 “空洞”?主键冲突、回滚、批量申请。

id 自增主键、c 唯一索引

insert into t values(null, 1, 1)show create table

AUTO_INCREMENT=2,下一次插入,生成 id=2。

表的结构定义存放在后缀名为 .frm 的文件中,不会保存自增值。

MyISAM 保存 数据文件 ; InnoDB 自增值保存 内存里(MySQL 8.0记录redo log 中) ,重启 恢复为重启前值

MySQL 5.7 前,没持久化。最大值 max(id)=10,AUTO_INCREMENT=11。删除 id=10 的行,AUTO_INCREMENT 还是 11。重启AUTO_INCREMENT 0。 重启修改AUTO_INCREMENT 。

id 被定义为 AUTO_INCREMENT,

1. 插入时 id 字段指定 0、null 或 未指定值 ,AUTO_INCREMENT 填自增字段 ;

2.  id 指定具体值, 用指定值 。

插入 X,自增值 Y。

1.   X(2)<Y(3),自增值不变 ;

2.  如果 (2)X≥Y(1),前自增值 改为新自增值 X

新自增值生成算法是 :自增的初始值auto_increment_offset(默认1)开始, 步长auto_increment_increment (默认1)为,持续叠加,第一个大于 X 值,作为新的自增值。

ps:用的 不全是默认值 。如双 M 主备 结构里 双写 ,auto_increment_increment= 2 ,让自增 id 都是 奇数 ,另一都是 偶数 , 避免主键冲突 。

插入的值 >= 当前自增值,新自增值就是“准备插入的值 +1”;否则不变。

两个参数都设置为 1 的时候,自增主键 id 却不能保证是连续的,这是什么原因呢?

已经有了 (1,1,1)   insert into t  values(null, 1, 1)

1.  写入一行(0,1,1)

2.  没有指定自增 id 的值, t 当前的自增值 2 ;

3.  改成 (2 ,1,1); 自增值改成 3

4.  已经存在 c=1 的记录,所以报 Duplicate key error,语句返回。

没有插入成功, 自增值不再改回去。不连续。

自增值不能回退:提升性能

两个并行事务,加锁顺序申请。

1.   A 申请到了 id=2, B  id=3, t 的自增值是 4,

2.   B 提交, A 出现唯一键冲突。

3.  如允许A 自增 id 回退, t 改回 2 ,问题:id=3 再申请到 id=3(已有) “主键冲突”

1. 申请前判断 ,存在跳过。成本高。因为,本来申请 id 是一个很快的 *** 作,现在还要再去主键索引树上判断 id 是否存在。

2. 锁范围扩大 ,事务完 提交 , 下一个再申请 自增 id。粒度太大,并发能力下降。

0: 5.0 策略, 语句结束释放 ;

1 (默认): 普通 insert ,自增锁 申请后释放 ;

                insert …select 批量插入, 语句结束释放 ;

2 :申请后释放锁

ps: 5.0 版本,自增锁, 语句级别 。申请 表 自增锁,结束释放, 影响并发度 。

数据的一致性

t1 插入4 行,创建相同结构表 t2,同时向 t2 插入。

如果 B 申请自增值后马上释放自增锁 ,情况:

B 先插入了两个记录,( 1 ,1,1)、( 2 ,2,2);

A 来申请自增 id 得到 id=3,插入了( 3 ,5,5);

B 插入两条记录 ( 4 ,3,3)、 (5 ,4,4)。

B 本身就没要求 t2 跟A 相同。如果binlog_format=statement,binlog 里id 连续。数据不一致。

问题原因:B 的 insert 语句,生成id 不连续。 statement 格式的 binlog 串行执行, 执行不出来 。

1.  批量插入 数据语句, 固定生成连续 id 值 。语句结束释放

2.  binlog如实记录进来,备库执行,不依赖于自增主键生成。innodb_autoinc_lock_mode = 2,binlog_format = row。 生产上,尤其 insert … select 批量插入数据时, 提升并发性,不会数据一致性。

批量插入数据包含:  insert … select 、 replace … select 和 load data 

普通 insert 多个 value 情况,innodb_autoinc_lock_mode 设置 1,精确计算出要多少个 id 的, 一次性申请,释放 。

1.  第一次申请自增 id,会分配 1 个;

2.  第二次申请自增 id,会分配 2 个;

3. 第三次申请自增 id,会分配 4 个; 依此类推,上一次的两倍

实际上t2 中插入 4 行, 分三次 ,1,第二次id=2 和 id=3, 第三次id=4 到 id=7。

 id=5 到 id=7 浪费掉

再执行 insert into t2  values(null, 5,5),实际上插入的数据就是( 8 ,5,5)。

自增值存储。

MyISAM 里,被写数据文件上。 InnoDB 中,记录内存的。重启前后不变。

自增值改变时机,回滚不能回收自增 id。

innodb_autoinc_lock_mode,控制自增值申请锁范围。并发性能考虑,设置为 2,binlog_format =row。

最后例子,执行 insert into t2(c,d) select c,d from t隔离级别是 可重复读 (repeatable

read),binlog_format=statement。 所有记录和间隙加锁 。为什么这么做?

如果 insert …select 有其他线程 *** 作原表,不会导致逻辑错误。如不加锁,就是快照读(执行期间,一致性视图是不会修改)。

不对t表所有记录和间隙加锁,,可重复读,其他提交t2看不到。但binlog=statement,备库或基于binlog恢复临时库t2看到,不一致。

自增id和写binlog是有先后顺序的。binlog=statement,A获取id=1,B获取id=2,B提交,写binlog,再A写binlog。

如果binlog重放,不会出现不一致,B的id为1,A的id为2的情况

因为binlog 记录 自增值语句 前 ,前面多一句, 指定“自增ID值多少” ,对应主库自增值


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/zaji/7103183.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-04-01
下一篇 2023-04-01

发表评论

登录后才能评论

评论列表(0条)

保存