Oracle 12c提供的Identity Column特性简化了自增字段的定义。
声明自增字段通常有3种常见的用法,以下三种方式都支持INSERT
语句中省略自增字段的插入,但有些许差别。
1. GENERATED [ALWAYS] AS IDENTITY
此时ALWAYS
关键字是可选的:
create table tb_test (
id number GENERATED ALWAYS AS IDENTITY,
name varchar2(100)
);
此时试图插入指定ID
字段的元组将报错,因为ID
字段总是自动生成的,例如,执行下述语句:
INSERT INTO TB_TEST VALUES(1, 'Zyon');
将报:
ORA-32795: 无法插入到“始终生成”身份列
2. GENERATED BY DEFAULT AS IDENTITY
create table tb_test2 (
id number GENERATED BY DEFAULT AS IDENTITY,
name varchar2(100)
);
此时可以插入指定ID
列的元组,但不能指定ID
列为NULL
。
如:下述语句正常执行
INSERT INTO TB_TEST2 VALUES(1, 'Zyon');
但如下语句将会报错:
INSERT INTO TB_TEST2 VALUES(NULL, 'Darren');
3. GENERATED BY DEFAULT ON NULL AS IDENTITY
create table tb_test3 (
id number GENERATED BY DEFAULT on null AS IDENTITY,
name varchar2(100)
);
此时可以插入指定ID
列的元组,也可以指定插入ID
列为NULL
。
如下语句正常执行:
INSERT INTO TB_TEST3 VALUES(NULL, 'Darren');
注意
当自增列还要求唯一时,上述情形2和情形3将带来一个问题。
上述情形2和情形3允许插入指定自增列的元组,同时自增列自增时维护着下一次将要生成的序列号。所以如果同时支持自增,又插入了某个指定序列号的元组,后续插入时如果忽略自增列,那么当自增序列号达到指定序列号时,将会出现插入失败的情形。
但是此时自增计数器继续增加,所以下一次忽略自增列的插入可能又恢复正常,除非下一个序列号也被指定插入占用了。
例如:
忽略ID
字段插入了9个元组(ID
取值1-9),突然指定ID
为10插入第10个元组,然后又忽略ID
字段插入时,因为10已经被占用,如果自增列要求唯一,将会出现冲突。
所以最好还是统一插入时SQL的形式,不要混用。
参考: