繁体   English   中英

在连续的数字序列和 LAST_INSERT_ID 中填充已删除的数字

[英]Fill deleted numbers in consecutive series of numbers and LAST_INSERT_ID

首先,我是一个业余和非英语母语人士,所以如果你对我有一点耐心,我将不胜感激;)

在这里尝试做两件事,我不确定我是否应该就此做两个问题,但由于这一切都与我的情况有关,我想在一个问题中全部说出来。

我正在制作一种会计软件,理论上供我个人使用。 我几乎所有对象都使用数据库生成的 auto_increment ID,但对于某些特定情况,我需要一个“并行”更开放的 ID,它不会是主键但可以由用户操作(是的,我读过很多关于“你不需要连续的主键”的问题,我理解并同意,但让我说这个列不会是主键,让我们称之为“人而不是计算机-专家友好 ID") 匹配这些条件:

  • 当没有给出参数时,Id 应该自动增加。
  • 当一个数字作为参数给出时,如果未被占用,则应使用该数字,如果被占用则抛出异常。
  • 应该询问用户他/她是否想通过DELETE s 和任何其他操作来填充缺失的 ID,因此如果用户“说是”,应该自动找到并使用最小的缺失 ID。

我在 c# 中“手动”执行此操作没有问题,但是有什么方法可以直接在 MySQL 中实现这样的功能吗? 我在 MySQL 文档中读到AUTO_INCREMENT确实满足我的前两个条件,但即使它默认填充缺失的已删除数字,我不确定,我不希望它默认这样做,我需要软件首先询问,或者至少根据用户预先建立的配置来执行。

因此,我认为我应该在 c# 中手动完成(至少是最后一部分,但我怀疑我将被迫完全完成),这带来了关于LAST_INSERT_ID的问题。

所以,MYSQL 文档说:

如果前面的语句返回错误,则 LAST_INSERT_ID() 的值未定义。 对于事务表,如果语句因错误而回滚,则 LAST_INSERT_ID() 的值未定义。 对于手动 ROLLBACK,LAST_INSERT_ID() 的值不会恢复到事务前的值; 它保持在回滚时的状态。

我知道如果之前的INSERT语句因任何原因失败, LAST_INSERT_ID()基本上是无用的。

如果是这种情况,有没有办法检索最后插入的 ID 以确保出现故障时的已知行为? 类似于INSERT失败时返回 0 或 SQL 异常? 如果没有其他方法,那么标准的做法是什么(我想MAX(Id)不会这样做),如果存在类似标准的方法……或者我应该立即停止尝试这样做并首先执行更新,检查是否一切正常,然后执行SELECT LAST_INSERT_ID

总结:

  • 有没有办法直接在 MySQL 中实现一列满足给定条件的连续数字?
  • LAST_INSERT_ID怎么回事? 我应该放弃并且不直接使用它吗?

情况 1,知道要插入AUTO_INCREMENT的 id

尊重 AI 不是所描述的 PK。

-- drop table a12b;
create table a12b
(   id varchar(100) primary key,
    ai_id int not null AUTO_INCREMENT,
    thing varchar(100) not null,
    key(ai_id)
);

insert a12b (id,thing) values ('a','fish'); -- ai_id=1
insert a12b (id,thing) values ('b','dog'); -- 2
insert a12b (id,thing) values ('b2','cat'); -- 3
delete from a12b where id='b';
insert a12b(id,ai_id,thing) values ('b',2,'dog with spots'); -- 2 ******** right here
insert a12b (id,thing) values ('z','goat'); -- 4

select * from a12b;
+----+-------+----------------+
| id | ai_id | thing          |
+----+-------+----------------+
| a  |     1 | fish           |
| b  |     2 | dog with spots |
| b2 |     3 | cat            |
| z  |     4 | goat           |
+----+-------+----------------+
4 rows in set (0.00 sec)

情况 2,有一个系统,您可以在某个时候删除行。 并希望稍后填补那些明确删除的空白: 在此处查看我的答案

情况 3(INNODB 到处散布着一堆空白):

这不是问题的一部分。 也许使用使用辅助表的左连接(至少对于整数而不是 varchars。但是我们又在谈论整数)。 如果您需要在不知情的情况下发现差距,请使用辅助表(装有数字)进行左连接。 我知道这听起来很蹩脚,但辅助表精益求精,可以完成工作。 以下将是一个辅助表: https : //stackoverflow.com/a/33666394

INNODB 间隙异常

使用上表的 4 行,继续:

insert a12b (id,thing) values ('z','goat'); -- oops, problem, failed, but AI is incremented behind the scene
insert a12b (id,thing) values ('z2','goat'); -- 6 (you now have a gap)

data:
+----+-------+----------------+
| id | ai_id | thing          |
+----+-------+----------------+
| a  |     1 | fish           |
| b  |     2 | dog with spots |
| b2 |     3 | cat            |
| z  |     4 | goat           |
| z2 |     6 | goat           |
+----+-------+----------------+

有很多方法可以产生差距。 看到这个那个

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM