休眠获取下一个带有字符串前缀的 ID



我有一个关于Hibernate中主密钥生成的问题。我正在努力维护现有的注册表系统。当前设计使用字符串作为主键。规则类似于"EXE" + max().下面是表格的外观。

+----------+---------------------------+----------------+
| ID       |   Email                   |   Name         |
+----------+---------------------------+----------------+
|EXE1      | email1@gmail.com          | Name 1         |
+----------+---------------------------+----------------+
|EXE5      | email5@gmail.com          | Name 5         |
+----------+---------------------------+----------------+
|EXE14     | email14@gmail.com         | Name 14        |
+----------+---------------------------+----------------+
|EXE15     | email15@gmail.com         | Name 15        |
+----------+---------------------------+----------------+

目前我使用以下代码生成 ID。

Long rowCount = (Long) getSession().createCriteria(Exemption168DB.class).setProjection(Projections.rowCount()).uniqueResult();
if(rowCount == null)
rowCount = 0L;
return String.format("%s%d", CommonConstant.EXEMPTION_KEY_PREFIX, rowCount + 1);

但问题是;它使用行计数来获取下一个序列数字。因此,在上述情况下,该方法将返回EXE5(此 ID 已存在于表中,因此引发异常),因为表中的行数为 4,然后递增 1。我需要的是EXE16.

任何帮助都非常感谢。另外,我们使用 Informix 作为数据库引擎。

正如我在两条评论中指出的,Informix 中可用的一种技术将使用触发器和 SERIAL 列。 另一种技术将使用 SEQUENCE 和存储过程。

下面是序列和存储过程的一些演示代码:

CREATE SEQUENCE registry_seq
INCREMENT BY 3
START WITH 37
MINVALUE 21
MAXVALUE 299
CYCLE;
CREATE PROCEDURE get_next_registry_id() RETURNING VARCHAR(10) AS registry_id;
DEFINE i INTEGER;
DEFINE r VARCHAR(10);
SELECT registry_seq.NEXTVAL INTO i FROM "informix".SysTables WHERE tabid = 1;
LET r = "EXE" || i;
RETURN r;
END PROCEDURE;
CREATE TEMP TABLE registry
(
id              VARCHAR(10) NOT NULL UNIQUE,
email           VARCHAR(64) NOT NULL UNIQUE,
name            VARCHAR(64) NOT NULL UNIQUE
);
INSERT INTO registry VALUES('EXE1', 'email1@gmail.com', 'Name 1');
INSERT INTO registry VALUES('EXE5', 'email5@gmail.com', 'Name 5');
INSERT INTO registry VALUES('EXE14', 'email14@gmail.com', 'Name 14');
INSERT INTO registry VALUES('EXE15', 'email15@gmail.com', 'Name 15');
INSERT INTO registry VALUES(get_next_registry_id(), 'email' || registry_seq.currval || '@example.com', 'User ID ' || registry_seq.currval);
INSERT INTO registry VALUES(get_next_registry_id(), 'email' || registry_seq.currval || '@example.com', 'User ID ' || registry_seq.currval);
INSERT INTO registry VALUES(get_next_registry_id(), 'email' || registry_seq.currval || '@example.com', 'User ID ' || registry_seq.currval);
SELECT * FROM registry ORDER BY id;

显然,您将为 CREATE SEQUENCE 语句选择不同的控件值。 这些对我的测试来说半方便地工作(从在不同的桌子上开始工作)。

FROM "informix".systables WHERE tabid = 1是用于选择单行数据的标准 Informix 习惯用法。 系统目录具有记录tabid1systables表。 在 Informix 的现代版本(意味着您应该运行的任何内容;可能有些人仍在运行旧版本)上,您可以从sysmaster:sysdual(或者,如果您真的很安全,sysmaster:"informix".sysdual)中进行选择,这是一个带有单列的单行表。

最终输出为:

EXE1    email1@gmail.com        Name 1
EXE14   email14@gmail.com       Name 14
EXE15   email15@gmail.com       Name 15
EXE37   email37@example.com     User ID 37
EXE40   email40@example.com     User ID 40
EXE43   email43@example.com     User ID 43
EXE5    email5@gmail.com        Name 5

请注意,字母数字 ID 的缺点之一是排序顺序不是数字,而是字典顺序。

创建自定义 Id 生成器类,用于查询上次插入的 id 并提取数字部分。 选择字符串长度等于 ID 的最大字符串长度的所有 id,按降序排序并将结果集限制为 1。然后像在问题中那样增加数字。

最新更新