在oracle下触发器实现主键自增

时间:2021-09-02 05:06:39

1. 利用序列产生主键值

序列(Sequence)是一种可以被多个用户使用的用于产生一系列唯一数字的数据库对象。序列定义存储在数据字典中,通过提供唯一数值的顺序表来简化程序设计工作,可以使用序列自动产生主键的键值。当一个序列第一次被查询调用时,它将返回一个预定值。在随后的每次查询中,序列将产生一个按指定的增量增长的值。序列可以循环,或者是连续增加的,直到指定的最大值为止。

创建序列语法如下:

create sequence [模式]序列名称[start with 起始数字] [increment by 增量]
[maxvalue 最大值|nomaxvalue][minvalue 最小值|nominva lue][cycle|nocycle]
[cache 数目|nocache][order|noorder]。

在Oracle数据库中创建序列,在使用sql语句向数据库中写入数据的时候,利用序列产生的唯一值,实现表中主键值自增。例如:

--首先建一个表TEST
create table TEST
(
NID int PRIMARY KEY,
test1 varchar2(20),
test2 varchar2(20)
);


-- 再建一个序列SEQ_TEST
create sequence SEQ_TEST
minvalue 1 --最小值
nomaxvalue --不设置最大值
start with 1 --从1开始计数
increment by 1 --每次加1
nocycle --一直累加,不循环
nocache;
--不建缓冲区

insert into test(nid,test1) values(SEQ_TEST.nextval, 'aaa');
insert into test(nid,test1) values(SEQ_TEST.nextval, 'bbb');
select * from test;

一个序列可以被多个用户共用,可以针对所有的表,因此对某个表而言产生的主键值唯一但是可能不连续

2.使用触发器实现自增

在oracle数据库中,没有象MySQL的Autoincrement一样自动增长的数据类型。在实现Oracle数据库字段自增功能时,利用DML触发器来完成。

触发器(trigger)是一些过程,当发生一个特定的数据库事件时就执行这些过程,可以使用触发器扩充引用的完整性。

DML即数据操纵语言,用于让用户或程序员使用,实现对数据库中数据的操作。基本的数据操作分成两类四种:检索(查询)和更新(插入、删除、修改)。

DML触发器即由DML语句激发,其触发事件包括insert(插入)、update(更新)和delete(删除)。无论哪种触发事件,都能为每种触发事件创建before触发器和after触发器。如可以在表上建立一个before insert 语句,表示在insert事件发生之前采取行动。

创建触发器的语法如下:

create[or replace]trigger 触发器名称 

{before|after|instead of}激发触发事件

referencing_clause

[WHEN trigger_condition]

[FOR EACH ROW]

referencing_clause用来引用正在处于修改状态下的行中的数据,如果在WHEN子句中指定trigger_condition的话,则首先对该条件求值。触发器主体只有在该条件为真值时才运行。利用触发器与序列相结合,可以实现在进行DML操作的时候,使表中主键值自动增加。其实现步骤如下:

--首先建一个表TEST
create table TEST
(
NID int PRIMARY KEY,
test1 varchar2(20),
test2 varchar2(20)
);


-- 再建一个序列SEQ_TEST
create sequence SEQ_TEST
minvalue 1 --最小值
nomaxvalue --不设置最大值
start with 1 --从1开始计数
increment by 1 --每次加1
nocycle --一直累加,不循环
nocache;
--不建缓冲区

CREATE OR REPLACE TRIGGER tg_test
BEFORE INSERT ON test FOR EACH ROW WHEN (new.nid is null)
begin
select seq_test.nextval into:new.nid from dual;

end;

下面是测试:

insert into test(nid,test1) values(6,'aaa');
insert into test(test1) values('bbb');
select * from test;

如果没有对WHEN (new.nid is null)的判断,则不管有没有指定nid都会被替换为序列的值。