我在使用 IBM DB2 的自动增量列时遇到了一些问题。起初,我的所有列都定义为始终生成,但由于我在使用“db2 import ...”命令时遇到了麻烦,所以我将它们更改为默认生成。这是必要的,因为我需要 ID 保持一致,因为其他表会引用它们。因此使用“db2 import ... modified byidentityignore ...”不是一个选项。
当我现在导入数据时,ID 已正确插入,但每次执行此操作时,我都必须记住通过获取最高 Id+1 为自动增量列设置一个新的开始,并像这样更改该列:
SELECT MAX(mycolumn)+ 1 FROM mytable;
ALTER TABLE mytable ALTER COLUMN mycolumn RESTART WITH <above_result>;
如果我忘记了这一点,Insert-Statement 将失败并出现重复 PK 错误,因为自动增量列是主键。
所以我的问题是:有没有办法找到自动增量列的下一个值,这样我就可以编写语句来检查,如果该值小于 SELECT MAX 并且需要设置?
或者:这整件事难道不像我想象的那么复杂吗?我能否以某种方式导入数据,保留 ID,并使自动增量列仍按预期工作?
答案1
Tim,我遇到了同样的问题,需要将身份重新启动为下一个值。我使用的是 db2v9.1。
不幸的是,没有办法自动指定下一个值。根据 DB2 文档,该值应为“数字常量”。因此,我不得不执行 select max(id),获取该值并在 alter..restart stmt 中手动替换它。
我不记得我是否尝试过这个 - 但你可以编写一个 sp,其中 max(id) 设置在变量中,并在 alter...restart stmt 中分配该变量。(我无法尝试,因为我不再有权访问任何 db2 数据库)。但我怀疑它是否会起作用。(如果它有效,请告诉我 :))
DB2 参考:
RESTART 或 RESTART WITH 数字常量
重置与标识列关联的序列的状态。如果未指定 WITH numeric-constant,则标识列的序列将从最初创建标识列时隐式或显式指定的值重新启动为起始值。该列必须存在于指定的表中 (SQLSTATE 42703),并且必须已使用 IDENTITY 属性定义 (SQLSTATE 42837)。RESTART 不会更改原始 START WITH 值。
数字常量是精确的数字常量,可以是分配给此列的任何正值或负值(SQLSTATE 42815),小数点右侧不存在非零数字(SQLSTATE 428FA)。数字常量将用作该列的下一个值。
答案2
不能在 DB2 中使用序列号然后使用序列。nextval() 吗?
DB2 中自动生成序列值的链接: http://www.ibm.com/developerworks/data/library/techarticle/0302fielding/0302fielding.html
答案3
您的问题现在可能已经解决了,但对于所有遇到同样问题的人:我可以提供一种解决方案,用于获取所有表的 max_id +1。唯一的限制是,源列必须是自动增量列 (GENERATED= 'D')。结果集可以通过 clp 或其他查询工具执行。
SELECT 'alter table ' || rtrim(TABSCHEMA) || '.' || TABNAME || ' alter column ' || COLNAME || ' restart with ' ||
cast((case when (trim ((replace ( HIGH2KEY, '+' , '' )))) = '' then 0 else integer(trim ((replace ( HIGH2KEY, '+' , '' ))))+2 end) as varchar(20))
|| ' ;'
FROM SYSCAT.COLUMNS
WHERE GENERATED = 'D'
答案4
- 如果你手动导入该列的数据,为什么还要考虑使用自动增量?这个自动增量根本没用,
- 在这里的类似应用中,我编写了一个 SP - a. 调用 admin_cmd - 加载选项 b. 选择 max(id) - 并将它们与当前时间戳一起加载到日志表中只需查询日志表并使用 bash/windows 脚本进行更新即可。很简单....