我有一个关于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 习惯用法。 系统目录具有记录tabid
1
的systables
表。 在 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。然后像在问题中那样增加数字。