阿里云-云小站(无限量代金券发放中)
【腾讯云】云服务器、云数据库、COS、CDN、短信等热卖云产品特惠抢购

Oracle触发器详细讲解

356次阅读
没有评论

共计 4693 个字符,预计需要花费 12 分钟才能阅读完成。

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

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

Oracle 触发器详细讲解Oracle 触发器详细讲解

一、触发器概念

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;

Oracle 触发器详细讲解

这里看到在创建触发器以后如果仍然使用 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;

Oracle 触发器详细讲解

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

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;

Oracle 触发器详细讲解

这里可以看到在创建触发器时,用到了 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′;

Oracle 触发器详细讲解

这里可以看到当改变数据时会触发触发器错误,对表中某一个字段的修改用 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;

Oracle 触发器详细讲解

select * from emp_new;

Oracle 触发器详细讲解

这里参照完整新指具有主从关系的多个表,当更新主表主键时需要更新从表的相关数据。

3、替代触发器:

这里先讲另一个概念:带有 with check option 的视图:

如果视图的定义包括条件(如 where 子句)并且任何应用于该视图的 INSERT 或 UPDATE 语句都应包括该条件,则必须使用 WITH CHECK OPTION 定义该视图。

Example:

CREATE VIEW emp_view
(ename,empno)
AS SELECT ename,empno FROM emp
WHERE deptno=20
WITH CHECK OPTION;

这里有个条件部门号为 20,则任何修改这个视图的语句都必须针对的是 20 号部门的员工。

继续替代触发器的概念:关键字 insteadof,主要针对一些复杂的视图,因为级联表所产生的视图不可以使用 update,insert,delete 等关键字,没有 before,after 等关键字,并且不可以建立在 with check option 选项的视图上,比如新建一个 emp 表和 dept 表的级联视图,则不可以向其中添加数据,现在通过触发器解决:

Example:

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

CREATE TABLE emp_new
AS
SELECT * FROM emp;
CREATE TABLE dept_new
AS
SELECT * FROM dept;CREATE VIEW emp_dept
AS
SELECT d.deptno,d.dname,e.empno,e.ename
FROM dept_new d,emp_new e
WHERE d.deptno=e.deptno;

这里 scott 用户需要先通过 sysdba 授权才能建立视图:

grant create view to scott;

CREATE OR REPLACE TRIGGER insteadof_trigger
INSTEAD OF INSERT ON emp_dept
FOR EACH ROW
DECLARE
    v_temp INT;
BEGIN
    SELECT COUNT(*) INTO v_temp FROM dept_new WHERE deptno=:new.deptno;
    IF v_temp=0 THEN
      INSERT INTO dept_new(deptno,dname) VALUES(:new.deptno,:new.dname);
    END IF;
    SELECT COUNT(*) INTO v_temp FROM emp_new WHERE empno=:new.empno;
    IF v_temp=0 THEN
      INSERT INTO emp_new(deptno,empno,ename) VALUES(:new.deptno,:new.empno,:new.ename);
    END IF;
END;

INSERT INTO emp_dept values(15,’HUMANRESOURCE’,7999,’LEAF’);select * from emp_new;

Oracle 触发器详细讲解

select * from dept_new;

Oracle 触发器详细讲解

这里触发器中当对视图进行 insert 时,会对相应的 emp_new 和 dept_new 进行修改,也就做到了对复杂视图的修改。

4、系统触发器:

顾名思义,由系统触发器所触发的事件,常用的系统事件 startup,shutdown,db_roll_change,server error 等。

Example: 记录启动数据库时的事件以及时间。

此处因为是系统触发器,所以需要用 sysdba 的权限登陆。

CREATE TABLE event_table(event VARCHAR2(50),event_time DATE);CREATE OR REPLACE TRIGGER event_trigger
AFTER STARTUP ON DATABASE
BEGIN
  INSERT INTO event_table VALUES(ora_sysevent,sysdate);
END;

Oracle 触发器详细讲解

select * from event_table;

Oracle 触发器详细讲解

三、触发器的综合实例

Example:做一个日志用来记录 scott 用户的一些操作:

首先在 sysdba 权限下建立日志表,序列,触发器:

CREATE TABLE object_log(
logid NUMBER CONSTRAINT pk_logid PRIMARY KEY,
operatedate DATE NOT NULL,
objecttype VARCHAR2(50) NOT NULL,
objectowner VARCHAR2(50) NOT NULL
);CREATE SEQUENCE obj_log_seq;CREATE OR REPLACE TRIGGER object_trigger
AFTER CREATE OR DROP OR ALTER ON DATABASE
BEGIN
  INSERT INTO object_log VALUES(obj_log_seq.nextval,sysdate,ora_dict_obj_type,ora_dict_obj_owner);
END;

在 scott 用户下随便创建个东西:

CREATE SEQUENCE my_seq;

回到 sysdba 权限下查看日志表中是否有对应的记录:

SELECT * FROM object_log;

Oracle 触发器详细讲解

发现有数据,说明一个日志表成功做好,监视一些用户操作的触发器就做好了。

至此,触发器全部说明完毕,不足之处还请评论说明,谢谢。

更多 Oracle 相关信息见 Oracle 专题页面 https://www.linuxidc.com/topicnews.aspx?tid=12

正文完
星哥玩云-微信公众号
post-qrcode
 0
星锅
版权声明:本站原创文章,由 星锅 于2022-01-22发表,共计4693字。
转载说明:除特殊说明外本站文章皆由CC-4.0协议发布,转载请注明出处。
【腾讯云】推广者专属福利,新客户无门槛领取总价值高达2860元代金券,每种代金券限量500张,先到先得。
阿里云-最新活动爆款每日限量供应
评论(没有评论)
验证码
【腾讯云】云服务器、云数据库、COS、CDN、短信等云产品特惠热卖中

星哥玩云

星哥玩云
星哥玩云
分享互联网知识
用户数
4
文章数
19348
评论数
4
阅读量
7805451
文章搜索
热门文章
开发者必备神器:阿里云 Qoder CLI 全面解析与上手指南

开发者必备神器:阿里云 Qoder CLI 全面解析与上手指南

开发者必备神器:阿里云 Qoder CLI 全面解析与上手指南 大家好,我是星哥。之前介绍了腾讯云的 Code...
星哥带你玩飞牛NAS-6:抖音视频同步工具,视频下载自动下载保存

星哥带你玩飞牛NAS-6:抖音视频同步工具,视频下载自动下载保存

星哥带你玩飞牛 NAS-6:抖音视频同步工具,视频下载自动下载保存 前言 各位玩 NAS 的朋友好,我是星哥!...
云服务器部署服务器面板1Panel:小白轻松构建Web服务与面板加固指南

云服务器部署服务器面板1Panel:小白轻松构建Web服务与面板加固指南

云服务器部署服务器面板 1Panel:小白轻松构建 Web 服务与面板加固指南 哈喽,我是星哥,经常有人问我不...
我把用了20年的360安全卫士卸载了

我把用了20年的360安全卫士卸载了

我把用了 20 年的 360 安全卫士卸载了 是的,正如标题你看到的。 原因 偷摸安装自家的软件 莫名其妙安装...
星哥带你玩飞牛NAS-3:安装飞牛NAS后的很有必要的操作

星哥带你玩飞牛NAS-3:安装飞牛NAS后的很有必要的操作

星哥带你玩飞牛 NAS-3:安装飞牛 NAS 后的很有必要的操作 前言 如果你已经有了飞牛 NAS 系统,之前...
阿里云CDN
阿里云CDN-提高用户访问的响应速度和成功率
随机文章
我用AI做了一个1978年至2019年中国大陆企业注册的查询网站

我用AI做了一个1978年至2019年中国大陆企业注册的查询网站

我用 AI 做了一个 1978 年至 2019 年中国大陆企业注册的查询网站 最近星哥在 GitHub 上偶然...
星哥带你玩飞牛NAS硬件 01:捡垃圾的最爱双盘,暴风二期矿渣为何成不老神话?

星哥带你玩飞牛NAS硬件 01:捡垃圾的最爱双盘,暴风二期矿渣为何成不老神话?

星哥带你玩飞牛 NAS 硬件 01:捡垃圾的最爱双盘,暴风二期矿渣为何成不老神话? 前言 在选择 NAS 用预...
从“纸堆”到“电子化”文档:用这个开源系统打造你的智能文档管理系统

从“纸堆”到“电子化”文档:用这个开源系统打造你的智能文档管理系统

从“纸堆”到“电子化”文档:用这个开源系统打造你的智能文档管理系统 大家好,我是星哥。公司的项目文档存了一堆 ...
4盘位、4K输出、J3455、遥控,NAS硬件入门性价比之王

4盘位、4K输出、J3455、遥控,NAS硬件入门性价比之王

  4 盘位、4K 输出、J3455、遥控,NAS 硬件入门性价比之王 开篇 在 NAS 市场中,威...
自己手撸一个AI智能体—跟创业大佬对话

自己手撸一个AI智能体—跟创业大佬对话

自己手撸一个 AI 智能体 — 跟创业大佬对话 前言 智能体(Agent)已经成为创业者和技术人绕...

免费图片视频管理工具让灵感库告别混乱

一言一句话
-「
手气不错
让微信公众号成为 AI 智能体:从内容沉淀到智能问答的一次升级

让微信公众号成为 AI 智能体:从内容沉淀到智能问答的一次升级

让微信公众号成为 AI 智能体:从内容沉淀到智能问答的一次升级 大家好,我是星哥,之前写了一篇文章 自己手撸一...
星哥带你玩飞牛NAS-5:飞牛NAS中的Docker功能介绍

星哥带你玩飞牛NAS-5:飞牛NAS中的Docker功能介绍

星哥带你玩飞牛 NAS-5:飞牛 NAS 中的 Docker 功能介绍 大家好,我是星哥,今天给大家带来如何在...
每年0.99刀,拿下你的第一个顶级域名,详细注册使用

每年0.99刀,拿下你的第一个顶级域名,详细注册使用

每年 0.99 刀,拿下你的第一个顶级域名,详细注册使用 前言 作为长期折腾云服务、域名建站的老玩家,星哥一直...
每天一个好玩的网站-手机博物馆-CHAZ 3D Experience

每天一个好玩的网站-手机博物馆-CHAZ 3D Experience

每天一个好玩的网站 - 手机博物馆 -CHAZ 3D Experience 一句话介绍:一个用 3D 方式重温...
星哥带你玩飞牛 NAS-10:备份微信聊天记录、数据到你的NAS中!

星哥带你玩飞牛 NAS-10:备份微信聊天记录、数据到你的NAS中!

星哥带你玩飞牛 NAS-10:备份微信聊天记录、数据到你的 NAS 中! 大家对「数据安全感」的需求越来越高 ...