Oracle数据库之PL/SQL触发器

简介: 1. 介绍 触发器(trigger)是数据库提供给程序员和数据分析员来保证数据完整性的一种方法,它是与表事件相关的特殊的存储过程,它的执行不是由程序调用,也不是手工启动,而是由事件来触发,比如当对一个表进行操作(insert,delete,update)时就会激活它执行。

1. 介绍

触发器(trigger)是数据库提供给程序员和数据分析员来保证数据完整性的一种方法,它是与表事件相关的特殊的存储过程,它的执行不是由程序调用,也不是手工启动,而是由事件来触发,比如当对一个表进行操作(insert,delete,update)时就会激活它执行。触发器经常用于加强数据的完整性约束和业务规则等。

ORACLE触发器有三种类型,分别是:DML触发器、替代触发器和系统触发器。

DML触发器

顾名思义,DML触发器是由DML语句触发的。例如数据库的INSERT、UPDATE、DELETE操作都可以触发该类型的触发器。它们可以在这些语句之前或之后触发,或者在行级上触发(就是说对于每个受影响的行都触发一次)。

替代触发器

替代触发器只能使用在视图上,与DML不同的是,DML触发器是运行在DML之外的,而替代触发器是代替激发它的DML语句运行。替代触发器是行触发器。

系统触发器

这种触发器是发生在如数据库启动或关闭等系统事件时,不是在执行DML语句时发生,当然也可以在DDL时触发。

触发器功能强大,轻松可靠地实现许多复杂的功能,但是我们也应该慎用。为什么又要慎用呢?触发器本身没有过错,但如果我们滥用,会造成数据库及应用程序的维护困难。在数据库操作中,我们可以通过关系、触发器、存储过程、应用程序等来实现数据操作,同时约束、缺省值也是保证数据完整性的重要保障。如果我们对触发器过分的依赖,势必影响数据库的结构,同时增加了维护的复杂程度。

2. 触发器组成

触发器主要由以下几个要素组成:

  1. 触发事件:引起触发器被触发的事件。
  2. 触发时间:触发器是在触发事件发生之前(BEFORE)还是之后(AFTER)触发,也就是触发事件和该触发器的操作顺序。
  3. 触发操作:触发器被触发之后的目的和意图,是触发器本身要做的事情。
  4. 触发对象:包括表、视图、模式、数据库。只有在这些对象上发生了符合触发条件的触发事件,才会执行触发操作。
  5. 触发条件:由WHEN子句指定一个逻辑表达式。只有当该表达式的值为TRUE时,遇到触发事件才会自动执行触发器,使其执行触发操作。
  6. 触发频率:说明触发器内定义的动作被执行的频率。即语句级(STATEMENT)触发器和行级(ROW)触发器: 
    语句级(STATEMENT)触发器:是指当某触发事件发生时,该触发器只执行一次; 
    行级(ROW)触发器:是指当某触发事件发生时,对受到该操作影响的每一行数据,触发器都单独执行一次。

3. 创建触发器

语法:

 1 CREATE [ OR REPLACE ] TRIGGER plsql_trigger_source 

 1 plsql_trigger_source ::========
 2 
 3 [schema.] trigger_name
 4   { simple_dml_trigger
 5   | instead_of_dml_trigger
 6   | compound_dml_trigger
 7   | system_trigger
 8   }
 9 
10 simple_dml_trigger ::========
11 
12 { BEFORE | AFTER } dml_event_clause [ referencing_clause ] [ FOR EACH ROW ]
13   [ trigger_edition_clause ] [ trigger_ordering_clause ]
14     [ ENABLE | DISABLE ] [ WHEN ( condition ) ] trigger_body
15 
16 instead_of_dml_trigger ::=
17 
18 INSTEAD OF { DELETE | INSERT | UPDATE } [ OR { DELETE | INSERT | UPDATE } ]...
19 ON [ NESTED TABLE nested_table_column OF ] [ schema. ] noneditioning_view
20 [ referencing_clause ] [ FOR EACH ROW ]
21 [ trigger_edition_clause ] [ trigger_ordering_clause ]
22 [ ENABLE | DISABLE ] trigger_body
23 
24 system_trigger ::=======
25 
26 { BEFORE | AFTER | INSTEAD OF }
27 { ddl_event [OR ddl_event]...
28 | database_event [OR database_event]...
29 }
30 ON { [schema.] SCHEMA
31    | DATABASE
32    }
33 [ trigger_ordering clause ]
34 
35 dml_event_clause ::============
36 
37 { DELETE | INSERT | UPDATE [ OF column [, column ]... ] }
38 [ OR { DELETE | INSERT | UPDATE [ OF column [, column]... ] }...
39 ON [ schema.] { table | view }
40 
41 referencing_clause ::==========
42 
43 REFERENCING
44  { OLD [ AS ] old
45  | NEW [ AS ] new
46  | PARENT [ AS ] parent
47  }...
48 
49 trigger_body ::=======
50 
51 { plsql_block | CALL routine_clause }

 

 

说明:

BEFORE和AFTER指出触发器的触发时间分别为前触发和后触发方式,前触发是在执行触发事件之前触发当前所创建的触发器,后触发是在执行触发事件之后触发当前所创建的触发器。

REFERENCING子句说明相关名称,在行触发器的PL/SQL块和WHEN子句中可以使用相关名称参照当前的新、旧列值,默认的相关名称为OLD和NEW。触发器的PL/SQL块中应用相关名称时,必须在它们之前加冒号(:),但在WHEN子句中则不能加冒号。

NEW只在UPDATE、INSERT的DML触发器内可用,它包含了修改发生后被影响行的值。

OLD只在UPDATE、DELETE的DML触发器内可用,它包含了修改发生前被影响行的值。

FOR EACH ROW选项说明触发器为行触发器。行触发器和语句触发器的区别表现在:行触发器要求当一个DML语句操走影响数据库中的多行数据时,对于其中的每个数据行,只要它们符合触发约束条件,均激活一次触发器;而语句触发器将整个语句操作作为触发事件,当它符合约束条件时,激活一次触发器。当省略FOR EACH ROW 选项时,BEFORE和AFTER触发器为语句触发器,而INSTEAD OF触发器则只能为行触发器。

WHEN子句说明触发约束条件。Condition为一个逻辑表达时,其中必须包含相关名称,而不能包含查询语句,也不能调用PL/SQL函数。WHEN子句指定的触发约束条件只能用在BEFORE和AFTER行触发器中,不能用在INSTEAD OF行触发器和其它类型的触发器中。

INSTEAD OF选项(创建替代触发器)使ORACLE激活触发器,而不执行触发事件。只能对视图和对象视图建立INSTEAD OF触发器,而不能对表、模式和数据库建立INSTEAD OF触发器。

ddl_event:一个或多个DDL事件,事件间用OR分开。

database_event:一个或多个数据库事件,事件间用OR分开。

示例1,在插入数据时,自动使用序列编号:

1 CREATE OR REPLACE TRIGGER EMP_INSERT_ID
2 BEFORE INSERT ON employee FOR EACH ROW
3 BEGIN
4    SELECT SEQ_ID.NEXTVAL INTO :NEW.ID FROM DUAL;
5 END;
 1 示例2,在多表联接的视图中插入数据:
 2 
 3 -- 创建视图
 4 CREATE OR REPLACE VIEW vw_emp AS
 5 SELECT e.name ename, e.address, d.name dname
 6 FROM employee e, dept d
 7 WHERE e.did = d.id;
 8 
 9 -- 创建触发器
10 CREATE TRIGGER emp_insert_trigger
11    INSTEAD OF INSERT ON vw_emp
12 DECLARE
13    v_did dept.id%TYPE;
14 BEGIN
15    SELECT id INTO v_did FROM dept WHERE name = :NEW.dname;
16    INSERT INTO emp (name, address, did) VALUES (:NEW.ename, :NEW.address, v_did);
17 END emp_insert_trigger;

 

 

 1 示例3,创建实例启动触发器:
 2 
 3 -- 创建记录操作事件的表
 4 CREATE TABLE event_table(
 5    event VARCHAR2(50),
 6    time DATE
 7 );
 8 
 9 -- 创建触发器
10 CREATE OR REPLACE TRIGGER tr_startup
11    AFTER STARTUP
12    ON DATABASE
13 BEGIN
14    INSERT INTO event_table(event, time)
15     VALUES(ora_sysevent, SYSDATE);
16 END;

 

 

 DML触发器

DML触发器对我们开发人员来说是最常用的。DML触发器是由数据库的INSERT、UPDATE、DELETE操作触发,该类触发器可以在上述语句之前或之后执行,也可以每个受影响的行执行一次。

条件谓词:当在触发器中包含多个触发事件(INSERT、UPDATE、DELETE)的组合时,为了分别针对不同的事件进行不同的处理,需要使用ORACLE提供的条件谓词:

  1. INSERTING:当触发事件是INSERT时,取值为TRUE,否则为FALSE。
  2. UPDATING [(column_1,column_2,…,column_x)]:当触发事件是UPDATE时,如果修改了column_x列,则取值为TRUE,否则为FALSE。
  3. DELETING:当触发事件是DELETE时,则取值为TRUE,否则为FALSE。

示例:

 1 CREATE OR REPLACE TRIGGER emp_sal_trigger
 2    BEFORE UPDATE OF salary OR DELETE
 3    ON employee FOR EACH ROW
 4    WHEN (old.did = 1)
 5 BEGIN
 6   CASE
 7      WHEN UPDATING ('salary') THEN
 8         IF :NEW.salary < :old.salary THEN
 9            RAISE_APPLICATION_ERROR(-20001, '部门1的员工工资不能降');
10         END IF;
11      WHEN DELETING THEN
12           RAISE_APPLICATION_ERROR(-20002, '不能删除部门1的员工记录');
13   END CASE;
14 END emp_sal_trigger;

 

5. 替代触发器

INSTEAD OF用于对视图的DML触发,由于视图有可能是由多个表联结(JOIN)而成,因而并非所有的视图都是可更新的,但可以按照所需的方式执行更新。

创建INSTEAD OF触发器需要注意以下几点:

  1. 只能被创建在视图上,并且该视图没有指定WITH CHECK OPTION选项。
  2. 不能指定BEFORE或AFTER选项。
  3. FOR EACH ROW子句是可选的。
  4. 没有必要在针对一个表的视图上创建INSTEAD OF触发器,只要创建DML触发器就可以了。

示例:

1 CREATE OR REPLACE TRIGGER emp_delete_trigger
2    INSTEAD OF DELETE ON vw_emp FOR EACH ROW
3 DECLARE
4    v_did dept.id%TYPE;
5 BEGIN
6    SELEC id INTO v_did FROM dept WHERE name=:OLD.dname;
7    DELETE FROM employee WHERE did= v_did;
8 END emp_delete_trigger;

 

 

6. 系统触发器

系统触发器可以在DDL或数据库系统上被触发,数据库系统事件包括数据库服务器的启动或关闭,用户的登录与退出、数据库服务错误等。

系统事件触发器既可以建立在一个模式上,又可以建立在整个数据库上。当建立在模式(SCHEMA)之上时,只有模式所指定用户的DDL操作和它们所导致的错误才激活触发器,默认时为当前用户模式。当建立在数据库(DATABASE)之上时,该数据库所有用户的DDL操作和他们所导致的错误,以及数据库的启动和关闭均可激活触发器。

系统触发器的种类和事件出现的时机:

 

 1 -- 创建记录用户登录注销日志的表
 2 CREATE TABLE log_on_off_log
 3 (user_name VARCHAR2(20),
 4  logon_date timestamp,
 5  logoff_date timestamp);
 6 
 7 -- 创建登录触发器
 8 CREATE OR REPLACE TRIGGER logon_trigger
 9    AFTER LOGON ON DATABASE
10 BEGIN
11    INSERT INTO log_on_off_log (user_name, logon_date) VALUES (ora_login_user, systimestamp);
12 END logon_trigger;
13 
14 -- 创建退出触发器
15 CREATE OR REPLACE TRIGGER logoff_trigger
16    BEFORE LOGOFF ON DATABASE
17 BEGIN
18    INSERT INTO log_on_off_log (user_name, logoff_date) VALUES (ora_login_user, systimestamp);
19 END logoff_trigger;

 

 

事件 触发时机 说明
STARTUP AFTER 启动数据库实例之后触发
SHUTDOWN BEFORE 关闭数据库实例之前触发
SERVERERROR AFTER 数据库服务器发生错误之后触发
LOGON AFTER 成功登录到数据库后触发
LOGOFF BEFORE 断开数据库连接之前触发
DDL BEFORE,AFTER 在执行大多数DDL语句之前、之后触发
CREATE / ALTER / DROP BEFORE,AFTER 在执行CREATE或ALTER或DROP语句创建数据库对象之前、之后触发
RENAME BEFORE,AFTER 执行RENAME语句更改数据库对象名称之前、之后触发
GRANT / REVOKE BEFORE,AFTER 执行GRANT语句授予权限或REVOKE撤销权限之前、之后触发
AUDIT / NOAUDIT BEFORE,AFTER 执行AUDIT或NOAUDIT进行审计或停止审计之前、之后触发

 

相关文章
|
5天前
|
SQL 存储 Oracle
Oracle的PL/SQL定义变量和常量:数据的稳定与灵动
【4月更文挑战第19天】在Oracle PL/SQL中,变量和常量扮演着数据存储的关键角色。变量是可变的“魔术盒”,用于存储程序运行时的动态数据,通过`DECLARE`定义,可在循环和条件判断中体现其灵活性。常量则是不可变的“固定牌”,一旦设定值便保持不变,用`CONSTANT`声明,提供程序稳定性和易维护性。通过 `%TYPE`、`NOT NULL`等特性,可以更高效地管理和控制变量与常量,提升代码质量。善用两者,能优化PL/SQL程序的结构和性能。
|
1天前
|
SQL Oracle 关系型数据库
sql语句创建数据库
在创建数据库之前,请确保你有足够的权限,并且已经考虑了数据库的安全性和性能需求。此外,不同的DBMS可能有特定的最佳实践和配置要求,因此建议查阅相关DBMS的官方文档以获取更详细和准确的信息。
|
1天前
|
SQL 缓存 数据库
sql 数据库优化
SQL数据库优化是一个复杂且关键的过程,涉及多个层面的技术和策略。以下是一些主要的优化建议: 查询语句优化: 避免全表扫描:在查询时,尽量使用索引来减少全表扫描,提高查询速度。 使用合适的子查询方式:子查询可能降低查询效率,但可以通过优化子查询的结构或使用连接(JOIN)替代子查询来提高性能。 简化查询语句:避免不必要的复杂查询,尽量使SQL语句简单明了。 使用EXISTS替代IN:在查询数据是否存在时,使用EXISTS通常比IN更快。 索引优化: 建立合适的索引:对于经常查询的列,如主键和外键,应创建相应的索引。同时,考虑使用覆盖索引来进一步提高性能。 避免过多的索引:虽然索引可以提高查询
|
5天前
|
缓存 Oracle 关系型数据库
Oracle中的触发器与序列:自增列的魔法组合
【4月更文挑战第19天】Oracle数据库中,通过触发器和序列的组合可实现自增列功能。序列充当自动计数器,提供递增数值,而触发器则在插入新记录时自动分配序列值。创建序列如`CREATE SEQUENCE emp_seq START WITH 1 INCREMENT BY 1 NOCACHE`,然后创建触发器`TRIGGER trg_employees_before_insert`在`BEFORE INSERT`时将序列的下一个值赋予`employee_id`。这种方式使得在插入记录时无需手动设置ID,提高了效率。但使用时需注意序列状态、并发性和性能优化。
|
5天前
|
SQL Oracle 关系型数据库
Oracle的PL/SQL游标属性:数据的“导航仪”与“仪表盘”
【4月更文挑战第19天】Oracle PL/SQL游标属性如同车辆的导航仪和仪表盘,提供丰富信息和控制。 `%FOUND`和`%NOTFOUND`指示数据读取状态,`%ROWCOUNT`记录处理行数,`%ISOPEN`显示游标状态。还有`%BULK_ROWCOUNT`和`%BULK_EXCEPTIONS`增强处理灵活性。通过实例展示了如何在数据处理中利用这些属性监控和控制流程,提高效率和准确性。掌握游标属性是提升数据处理能力的关键。
|
5天前
|
SQL Oracle 安全
Oracle的PL/SQL循环语句:数据的“旋转木马”与“无限之旅”
【4月更文挑战第19天】Oracle PL/SQL中的循环语句(LOOP、EXIT WHEN、FOR、WHILE)是处理数据的关键工具,用于批量操作、报表生成和复杂业务逻辑。LOOP提供无限循环,可通过EXIT WHEN设定退出条件;FOR循环适用于固定次数迭代,WHILE循环基于条件判断执行。有效使用循环能提高效率,但需注意避免无限循环和优化大数据处理性能。掌握循环语句,将使数据处理更加高效和便捷。
|
5天前
|
SQL Oracle 关系型数据库
Oracle的PL/SQL条件控制:数据的“红绿灯”与“分岔路”
【4月更文挑战第19天】在Oracle PL/SQL中,IF语句与CASE语句扮演着数据流程控制的关键角色。IF语句如红绿灯,依据条件决定程序执行路径;ELSE和ELSIF提供多分支逻辑。CASE语句则是分岔路,按表达式值选择执行路径。这些条件控制语句在数据验证、错误处理和业务逻辑中不可或缺,通过巧妙运用能实现高效程序逻辑,保障数据正确流转,支持企业业务发展。理解并熟练掌握这些语句的使用是成为合格数据管理员的重要一环。
|
5天前
|
SQL Oracle 关系型数据库
Oracle的PL/SQL表达式:数据的魔法公式
【4月更文挑战第19天】探索Oracle PL/SQL表达式,体验数据的魔法公式。表达式结合常量、变量、运算符和函数,用于数据运算与转换。算术运算符处理数值计算,比较运算符执行数据比较,内置函数如TO_CHAR、ROUND和SUBSTR提供多样化操作。条件表达式如CASE和NULLIF实现灵活逻辑判断。广泛应用于SQL查询和PL/SQL程序,助你驾驭数据,揭示其背后的规律与秘密,成为数据魔法师。
|
1月前
|
关系型数据库 MySQL
Mysql基础第二十八天,使用触发器
Mysql基础第二十八天,使用触发器
28 0
Mysql基础第二十八天,使用触发器
|
1月前
|
存储 SQL 关系型数据库
【MySQL 数据库】10、MySQL 的触发器
【MySQL 数据库】10、MySQL 的触发器
20 0

热门文章

最新文章

推荐镜像

更多