Chinaunix首页 | 论坛 | 博客
  • 博客访问: 287797
  • 博文数量: 72
  • 博客积分: 2387
  • 博客等级: 大尉
  • 技术积分: 720
  • 用 户 组: 普通用户
  • 注册时间: 2010-08-26 10:54
文章分类

全部博文(72)

文章存档

2012年(1)

2011年(1)

2010年(70)

分类: Oracle

2010-08-26 12:05:31

1、触发器的概念

触发器也是一种带名的PL/SQL块。触发器类似于过程和函数,因为它们都是拥有声明、执行和异常处理过程的带名PL/SQL块。与包类似,触发器必须存储在数据库中并且不能被块进行本地化声明。

对于触发器而言,当触发事件发生的时候就会显式地执行该触发器,并且触发器不接受参数。

创建触发器的语法如下:

CREATE [OR REPLACE] TRIGGER trigger_name

{BEFORE | AFTER | INSTEAD OF} triggering_event

[referencing_clause]

[WHEN trigger_condition]

[FOR EACH ROW]

Trigger_body;

其中referencing_clause子句的用途是通过一个不同的名称,引用当前正在被更新的记录行中的数据。WHEN子句中的trigger_condition—如果出现—就应该首先执行判断,只有当这个条件值为真的时候,才会执行触发器的主体代码。

2、DML触发器的激活顺序

1)执行before语句级触发器—如果存在这种触发器

2)对受该语句影响的每一行记录

执行before行级触发器—如果存在这种触发器

执行该语句本身

执行after行级触发器--如果存在这种触发器

3)执行after语句级触发器--如果存在这种触发器

同一种类型的触发器的点火次序没有经过定义。如果该次序很重要的话,那么建议将所有这些操作组合到一个触发器当中。

3、行级触发器中的关联标识符

触发器的激活语句每处理一行数据,行级触发器就会激活一次。可以在这种行级触发器内部,访问正被处理的记录行中的数据。这是通过两个关联标识符--:old和:new—实现的。关联标识符也是PL/SQL的一种特殊的绑定变量。标识符前面的冒号,既说明这二者都是绑定变量,同时也说明它们不是一般的PL/SQL变量。PL/SQL编译器会将它们看作下面这个类型的记录:

Triggering_table%ROWTYPE

其中triggering_table是在其上定义触发器的表名。于是,下面这种引用

:new.field

就只有当其中的field是该触发表中的字段名时才会有效。

触发语句

:old

:new

INSERT

未定义—所有字段均为NULL

触发语句完成的时候,要插入的值

UPDATE

更新以前相应记录行的原始值

触发语句完成的时候,要更新的值

DELETE

删除以前相应记录行的原始值

未定义—所有字段均为NULL

注意:INSERT语句上没有定义:old标识符,DELETE语句上也没有定义:new标识符。如果再INSERT语句上使用:old标识符,或者在DELETE语句上使用:new标识符,PL/SQL并不会产生错误,但是这两个字段值都会为NULL。

伪记录

虽然在语法构成上,会将:new和:old看作triggering_table%ROWTYPE类型的记录,但是,实际上它们并不是记录。因此,那些能够在记录上正常执行的操作,并不能在:new和:old上执行。例如,不能将它们作为一个整体进行赋值。只能对其中的各个字段分别赋值。

view plaincopy to clipboardprint?

CREATE OR REPLACE TRIGGER TempDelete

BEFORE DELETE ON temp_table

FOR EACH ROW

DECLARE

v_TempRec temp_table%ROWTYPE;

BEGIN

/* This is not a legal assignment, since :old is not truly

a record. */

v_TempRec := :old;

/* We can accomplish the same thing, however, by assigning

the fields individually. */

v_TempRec.char_col := :old.char_col;

v_TempRec.num_col := :old.num_col;

END TempDelete;

/

CREATE OR REPLACE TRIGGER TempDelete

BEFORE DELETE ON temp_table

FOR EACH ROW

DECLARE

v_TempRec temp_table%ROWTYPE;

BEGIN

/* This is not a legal assignment, since :old is not truly

a record. */

v_TempRec := :old;

/* We can accomplish the same thing, however, by assigning

the fields individually. */

v_TempRec.char_col := :old.char_col;

v_TempRec.num_col := :old.num_col;

END TempDelete;

/

REFERENCING子句

还可以使用REFERENCING子句,为:old和:new换一个不同的名称。该子句出现在触发事件以后,WHEN子句以前。其语法如下:

REFERENCING [OLD AS old_name] [NEW AS new_name]

在触发器主体中,可以使用:old_name和:new_name分别代替:old和:new。

注意,在REFERENCING子句中关联标识符都不带冒号。

如下面这个例子所示

view plaincopy to clipboardprint?

CREATE OR REPLACE TRIGGER GenerateAuthorID

BEFORE INSERT OR UPDATE ON authors

REFERENCING new AS new_author

FOR EACH ROW

BEGIN

/* Fill in the ID field of authors with the next value from

author_sequence. Since ID is a column in authors, :new.ID

is a valid reference. */

SELECT author_sequence.NEXTVAL

INTO :new_author.ID

FROM dual;

END GenerateAuthorID;

/

CREATE OR REPLACE TRIGGER GenerateAuthorID

BEFORE INSERT OR UPDATE ON authors

REFERENCING new AS new_author

FOR EACH ROW

BEGIN

/* Fill in the ID field of authors with the next value from

author_sequence. Since ID is a column in authors, :new.ID

is a valid reference. */

SELECT author_sequence.NEXTVAL

INTO :new_author.ID

FROM dual;

END GenerateAuthorID;

/

4、WHEN子句

WHEN子句只能在行级触发器中使用。如果在行级触发器的定义中给出了WHEN子句,触发器主体就只对满足WHEN所定义条件的那些记录行执行。WHEN子句的基本形式如下:

WHEN  trigger_condition

其中,trigger_condition是一个布尔表达式。每处理一行记录,都会重新判断该表达式的值。

也可以在trigger_condition内部使用:new和:old记录,但是与REFERENCING子句一样,在trigger_condition内部使用:new和:old时,不需要冒号。仅在触发器主体中才需要使用冒号。

阅读(986) | 评论(0) | 转发(0) |
0

上一篇:test

下一篇:我的trigger的例子

给主人留下些什么吧!~~