Oracle中触发器示例详解


本文整理自网络,侵删。

前言

在ORACLE系统里,触发器类似过程和函数,都有声明,执行和异常处理过程的PL/SQL块。

触发器类型

触发器在数据库里以独立的对象存储,它与存储过程和函数不同的是,存储过程与函数需要用户显示调用才执行,而触发器是由一个事件来启动运行。即触发器是当某个事件发生时自动地隐式运行。并且,触发器不能接收参数。所以运行触发器就叫触发或点火(firing)。ORACLE事件指的是对数据库的表进行的INSERT、UPDATE及DELETE操作或对视图进行类似的操作。ORACLE将触发器的功能扩展到了触发ORACLE,如数据库的启动与关闭等。所以触发器常用来完成由数据库的完整性约束难以完成的复杂业务规则的约束,或用来监视对数据库的各种操作,实现审计的功能。

开发中肯定会用到Oracle的触发器,本文进行详细讲解。

这里实例中用到的主要是Oracle中scott用户下的emp以及dept表,数据如下

一、触发器概念

1、概念:

触发器的本质是一个存储过程,顾名思义发生特定事件时Oracle会执行触发器中的代码。细分它的组成可以分为3个部分:第一部分在什么条件下触发器会执行,即触发器被触发的事件。第二部分在什么时间点执行触发器即触发器的发生事件例如before,after。第三部分触发器自身所要做的事情,就是触发器被触发以后具体想表达的事件,在begin和end之间的sql。

二、触发器的分类:

1、ddl触发器:即执行ddl操作后所触发的事件。

常用的ddl操作有:grant(授权),revoke(撤销授权),create(创建),drop(删除),alter(修改),comment(注释),audit(审核),rename(重命名)在进行具体实例以前先来讲解另一个概念:oracle中的user和schema:

user:oracle中的用户,拥有数据库的对象以及对数据库对象增删改查的权限。schema:该用户下所有数据库对象的集合Collection.类似于生活中房子schema和房子的拥有者user之间的关系,你是一个用户user你可以通过alter session查看别人的房子,但是你是否可以改变房子中的家具,要看这个房子的拥有者是否grant你这个权限,除非你是所有房子的最高权限人dba。

ddl Example:禁止scott用户的所有ddl操作

CREATE OR REPLACE TRIGGER scott_trigger
BEFORE DDL
ON SCHEMA
BEGIN
 RAISE_APPLICATION_ERROR(-20008,'禁止scott用户的所有ddl操作');
END;
create sequence myseq;

这里看到在创建触发器以后如果仍然使用ddl操作,便会报错。

2、dml触发器:基于dml操作的触发器,细分又可以分为行触发器和语句触发器。

A、语句触发器:dml操作可能会影响很多行,主要用于对数据的安全保护。

Example:禁止在周四,周五修改emp表数据

CREATE OR REPLACE TRIGGER emp_trigger
BEFORE UPDATE OR DELETE OR INSERT
ON emp
BEGIN
 IF to_char(sysdate,'day') IN ('星期四','星期五') THEN
 RAISE_APPLICATION_ERROR(-20008,'不允许在周四周五修改emp表');
 END IF;
END;
update emp set sal=800;

这里建立触发器以后,当你想改变所有人的工资时就会出触发器的错误,所有人的工资即表示会影响很多行。

B、行级触发器:针对需要操作的那一行,有关键词:for each row,用来

(1)实现数据的审计功能:

Example:做一个记录删除员工信息的表记录被删除员工的信息

这里为了不改变oracle中emp表的数据,新建一个emp_new表

create table emp_new
as
select * from emp;
create table emp_audit(name varchar2(10),delete_time Date);
CREATE OR REPLACE TRIGGER delete_trigger
AFTER DELETE ON emp_new
FOR EACH ROW
BEGIN
 INSERT INTO emp_audit values(:old.ename,sysdate);
END;
delete from emp_new where empno='7499';
select * from emp_audit;

这里可以看到在创建触发器时,用到了for each row关键词,:old.***用来表示更改以前的表中的数据,:new.***用来表示更改以后的数据,在删除数据以后在日志表就有对应的记录。

(2)实现数据完整性:

Example:要求员工涨工资后,不能低于原来的工资,所涨工资也不能高于原来的50%。

这里为了不改变oracle中emp表的数据,新建一个emp_new表

create table emp_new
as
select * from emp;
CREATE OR REPLACE TRIGGER emp_trigger
BEFORE UPDATE OF sal ON emp_new
FOR EACH ROW
WHEN (new.sal<old.sal OR new.sal>1.5*old.sal)
BEGIN
 RAISE_APPLICATION_ERROR(-20008,'工资只增不降,且涨幅不可大于50%');
END;
update emp_new set sal = 1.6*sal where empno='7788';

这里可以看到当改变数据时会触发触发器错误,对表中某一个字段的修改用UPDATE OF即可,另外如果new和old在PLSQL块的外部即BEGIN外面不可以加冒号。

(3)参照完整性:

Example:主要用于级联更新,如更新dept表中的deptno时,emp表的deptno也更新。

这里仍然新建2个表分别和emp表dept表的数据相同。

create table emp_new
as
select * from emp;
create table dept_new
as
select * from dept;
CREATE OR REPLACE TRIGGER cascade_trigger
AFTER UPDATE OF deptno ON dept_new
FOR EACH ROW
BEGIN
 UPDATE emp_new SET deptno=:new.deptno WHERE deptno=:old.deptno;
END;
update dept_new set deptno=15 where deptno=20;
select * from dept_new;

select * from emp_new;

阅读剩余部分

相关阅读 >>

oracle怎么查看用户对应的表空间

oracle例外用法实例详解

oracle别名使用要点小结

oracle查看用户是否被锁

jdbc连接oracle数据库功能示例

oracle转小写函数是什么

oracle 区块链表创建过程详解

oracle多表简单查询实例代码

oracle 数据库自动存储管理-安装配置

oracle查看和修改连接数(进程会话并发等等)

更多相关阅读请进入《oracle》频道 >>


数据库系统概念 第6版
书籍

数据库系统概念 第6版

机械工业出版社

本书主要讲述了数据模型、基于对象的数据库和XML、数据存储和查询、事务管理、体系结构等方面的内容。



打赏

取消

感谢您的支持,我会继续努力的!

扫码支持
扫码打赏,您说多少就多少

打开支付宝扫一扫,即可进行扫码打赏哦

分享从这里开始,精彩与您同在

评论

管理员已关闭评论功能...