版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领
文档简介
1、触发器使用教程和命名规范V1.0属性内容文档名称:触发器使用教程和命名规范文档版本号:V1.0文档编写日期:2006-12-4文档状态:修订稿作者:李振国发布日期:2006-12-5目 录触发器使用教程和命名规范1,触发器简介2,触发器示例3,触发器语法和功能4,例一:行级触发器之一5,例二:行级触发器之二6,例三:instead of触发器7,例四:语句级触发器之一8,例五:语句级触发器之二9,例六:用包封装触发器代码10,触发器命名规范1,触发器简介触发器(Trigger)是数据库对象的一种,编码方式类似存储过程,与某张表(Table)相关联,当有DML语句对表进行操作时,可以引
2、起触发器的执行,达到对插入记录一致性,正确性和规范性控制的目的。在当年C/S时代盛行的时候,由于客户端直接连接数据库,能保证数据库一致性的只有数据库本身,此时主键(Primary Key),外键(Foreign Key),约束(Constraint)和触发器成为必要的控制机制。而触发器的实现比较灵活,可编程性强,自然成为了最流行的控制机制。到了B/S时代,发展成4层架构,客户端不再能直接访问数据库,只有中间件才可以访问数据库。要控制数据库的一致性,既可以在中间件里控制,也可以在数据库端控制。很多的青睐Java的开发者,随之将数据库当成一个黑盒,把大多数的数据控制工作放在了Servlet中执行。
3、这样做,不需要了解太多的数据库知识,也减少了数据库编程的复杂性,但同时增加了Servlet编程的工作量。从架构设计来看,中间件的功能是检查业务正确性和执行业务逻辑,如果把数据的一致性检查放到中间件去做,需要在所有涉及到数据写入的地方进行数据一致性检查。由于数据库访问相对于中间件来说是远程调用,要编写统一的数据一致性检查代码并非易事,一般采用在多个地方的增加类似的检查步骤。一旦一致性检查过程发生调整,势必导致多个地方的修改,不仅增加工作量,而且无法保证每个检查步骤的正确性。触发器的应用,应该放在关键的,多方发起的,高频访问的数据表上,过多使用触发器,会增加数据库负担,降低数据库性能。而放弃使用触
4、发器,则会导致系统架构设计上的问题,影响系统的稳定性。2,触发器示例触发器代码类似存储过程,以PL/SQL脚本编写。下面是一个触发器的示例:新建员工工资表salarycreate table SALARY( EMPLOYEE_ID NUMBER, -员工ID MONTH VARCHAR2(6), -工资月份 AMOUNT NUMBER -工资金额)创建与salar
5、y关联的触发器salary_trg_rai1 Create or replace trigger salary_trg_rai2 After insert on salary3 For each row4 declare5 Begin6 Dbms_output.put_line(员工ID: | :new.employee_id);7 Dbms_output.
6、put_line(工资月份: | :new.month);8 Dbms_output.put_line(工资: | :new.amount);9 Dbms_output.put_line(触发器已被执行);10 End;打开一个SQL Window窗口(使用PL/SQL Developer工具),或在sqlplus中输入:Insert into salary(employee_id, month, amount) values(1, 200606, 10000);执行后可以在
7、sqlplus中,或在SQL Window窗口的Output中见到员工ID:1工资月份:200606工资:10000触发器已执行在代码的第一行,定义了数据库对象的类型是trigger,定义触发器的名称是salary_trg_rai第二行说明了这是一个after触发器,在DML操作实施之后执行。紧接着的insert说明了这是一个针对insert操作的触发器,每个对该表进行的insert操作都会执行这个触发器。第三行说明了这是一个针对行级的触发器,当插入的记录有n条时,在每一条插入操作时都会执行该触发器,总共执行n次。Declare后面跟的是本地变量定义部分,如果没有本地变量定义,此部分可以为空B
8、egin和end括起来的代码,是触发器的执行部分,一般会对插入记录进行一致性检查,在本例中打印了插入的记录和“触发器已执行”。其中:new对象表示了插入的记录,可以通过:new.column_name来引用记录的每个字段值3,触发器语法和功能触发器的语法如下CREATE OR REPLACE TRIGGER trigger_name<before | after | instead of> <insert | update | delete> ON table_nameFOR EACH ROWWHEN (condition)DECLAREBEGIN&
9、#160; -触发器代码END;Trigger_name是触发器的名称。<before | after | instead of>可以选择before或者after或instead of。Before表示在DML语句实施前执行触发器,而after表示在在dml语句实施之后执行触发器,instead of触发器用在对视图的更新上。<insert | update | delete>可以选择一个或多个DML语句,如果选择多个,则用or分开,如:insert or update。Table_name是触发器关联的表名
10、。FOR EACH ROW为可选项,如果注明了FOR EACH ROW,则说明了该触发器是一个行级的触发器,DML语句处理每条记录都会执行触发器;否则是一个语句级的触发器,每个DML语句触发一次。WHEN后跟的condition是触发器的响应条件,只对行级触发器有效,当操作的记录满足condition时,触发器才被执行,否则不执行。Condition中可以通过new对象和old对象(注意区别于前面的:new和:old,在代码中引用需要加上冒号)来引用操作的记录。触发器代码可以包括三种类型:未涉及数据库事务代码,涉及关联表(上文语法中的table_name)数据库事务代码,涉及除关联表之外数据库
11、事务代码。其中第一种类型代码只对数据进行简单运算和判断,没有DML语句,这种类型代码可以在所有的触发器中执行。第二种类型代码涉及到对关联表的数据操作,比如查询关联表的总记录数或者往关联表中插入一条记录,该类型代码只能在语句级触发器中使用,如果在行级触发器中使用,将会报ORA-04091错误。第三种类型代码涉及到除关联表之外的数据库事务,这种代码可以在所有触发器中使用。从触发器的功能上来看,可以分成3类:l 重写列(仅限于before触发器)l
12、 采取行动(任何触发器)l 拒绝事务(任何触发器)“重写列”用于对表字段的校验,当插入值为空或者插入值不符合要求,则触发器用缺省值或另外的值代替,在多数情况下与字段的default属性相同。这种功能只能在行级before触发器中执行。“采取行动”针对当前事务的特点,对相关表进行操作,比如根据当前表插入的记录更新其他表,银行中的总帐和分户帐间的总分关系就可以通过这种触发器功能来维护。“拒绝事务”用在对数据的合法性检验上,当更新的数据不
13、满足表或系统的一致性要求,则通过抛出异常的方式拒绝事务,在其上层的代码可以捕获这个异常并进行相应操作。下面将通过举例说明,在例子中将触发器主体的语法一一介绍,读者可以在例子中体会触发器的功能。4,例一:行级触发器之一CREATE OR REPLACE TRIGGER salary_raiuAFTER INSERT OR UPDATE OF amount ON salaryFOR EACH ROWBEGIN IF
14、;inserting THEN dbms_output.put_line(插入); ELSIF updating THENdbms_output.put_line(更新amount列); END IF;END;以上是一个after insert和after update的行级触发器。在第二行中of amount on salary的
15、意思是只有当amount列被更新时,update触发器才会有效。所以,以下语句将不会执行触发器:Update salary set month = 200601 where month = 200606;在触发器主体的if语句表达式中,inserting, updating和deleting可以用来区分当前是在做哪一种DML操作,可以作为把多个类似触发器合并在一个触发器中判别触发事件的属性。5,例二:行级触发器之二新建员工表employmentCREATE TABLE EMPLOYMENT( EMPLOYEE_ID NUMBER, -员工ID
16、MAXSALARY NUMBER -工资上限)插入两条记录Insert into employment values(1, 1000);Insert into employment values(2, 2000);CREATE OR REPLACE TRIGGER salary_raiuAFTER INSERT OR UPDATE OF amount ON salaryFOR EACH ROWWHEN (
17、0;NEW.amount >= 1000 AND (old.amount IS NULL OR OLD.amount <= 500)DECLARE v_maxsalary NUMBER;BEGIN SELECT maxsalary INTO v_maxsalary &
18、#160; FROM employment WHERE employee_id = :NEW.employee_id; IF :NEW.amount > v_maxsalary THEN raise_application_error(-20000, '工资超限');
19、 END IF;END;以上的例子引入了一个新的表employment,表中的maxsalary字段代表该员工每月所能分配的最高工资。下面的触发器根据插入或修改记录的employee_id,在employment表中查到该员工的每月最高工资,如果插入或修改后的amount超过这个值,则报错误。代码中的when子句表明了该触发器只针对修改或插入后的amount值超过1000,而修改前的amount值小于500的记录。New对象和old对象分别表示了操作前和操作后的记录对象。对于insert操作,由于当前操作记录无历史对象,所以old对象中所有属性是null;对于delete操
20、作,由于当前操作记录没有更新对象,所以new对象中所有属性也是null。但在这两种情况下,并不影响old和new对象的引用和在触发器主体中的使用,和普通的空值作同样的处理。在触发器主体中,先通过:new.employee_id,得到该员工的工资上限,然后在if语句中判断更新后的员工工资是否超限,如果超限则错误代码为-20000,错误信息为“工资超限”的自定义错误。其中的raise_application_error包含两个参数,前一个是自定义错误代码,后一个是自定义错误代码信息。其中自定义错误代码必须小于或等于-20000。执行完该语句后,一个异常被抛出,如果在上一层有exception子句,
21、该异常将被捕获。如下面代码:DECLARE code NUMBER; msg VARCHAR2(500);BEGIN INSERT INTO salary (employee_id, amount) VALUES (2, 5000);EXCEPTION WHEN OTHERS THEN
22、 code := SQLCODE; msg := substr(SQLERRM, 1, 500); dbms_output.put_line(code); dbms_output.put_line(msg);END;执行后,将在output中或者sqlplus窗口中见着以下信息:-2
23、0000ORA-20000: 工资超出限制ORA-06512: 在"SCOTT.SALARY_RAI", line 9ORA-04088: 触发器 'SCOTT.SALARY_RAI' 执行过程中出错这里的raise_application_error相当于拒绝了插入或者修改事务,当上层代码接受到这个异常后,判断该异常代码等于-20000,可以作出回滚事务或者继续其他事务的处理。以上两个例子中用到的inserting, updating, deleting和raise_application_error都是d
24、bms_standard包中的函数,具体的说明可以参照Oracle的帮助文档。create or replace package sys.dbms_standard is procedure raise_application_error(num binary_integer, msg varchar2, function inserting return boolean; function deleting return b
25、oolean; function updating return boolean; function updating (colnam varchar2) return boolean;end;对于before和after行级触发器,:new和:old对象的属性值都是一样的,主要是对于在Oracle约束(Constraint)之前或之后的执行触发器的选择。需要注意的是,可以在before行触发器中更改:new对象中的值,但是在after行触发器就不行。下面介绍一种instead of触发器,该触发器主要使
26、用在对视图的更新上,以下是instead of触发器的语法:CREATE OR REPLACE TRIGGER trigger_nameINSTEAD OF <insert | update | delete> ON view_nameFOR EACH ROWWHEN (condition)DECLAREBEGIN -触发器代码END;其他部分语法同前面所述的before和after语法是一样的,唯一不同的是在第二行用上了instead of关键字。对于普通的视图来说,进行insert等操作
27、是被禁止的,因为Oracle无法知道操作的字段具体是哪个表中的字段。但我们可以通过建立instead of触发器,在触发器主体中告诉Oracle应该更新,删除或者修改哪些表的哪部分字段。如:6,例三:instead of触发器新建视图CREATE VIEW employee_salary(employee_id, maxsalary, MONTH, amount) AS SELECT a.employee_id, a.maxsalary, b.MONTH, b.amountFROM employment a, salary
28、bWHERE a.employee_id = b.employee_id如果执行插入语句INSERT INTO employee_salary(employee_id, maxsalary, MONTH, amount)VALUES(10, 100000, '200606', 10000);系统会报错:ORA-01779:无法修改与非键值保存表对应的列我们可以通过建立以下的instead of存储过程,将插入视图的值分别插入到两个表中:create or replace trig
29、ger employee_salary_rii instead of insert on employee_salary for each ROWDECLARE v_cnt NUMBER;BEGIN -检查是否存在该员工信息 SELECT COUNT(*) INTO v_cnt
30、160; FROM employment WHERE employee_id = :NEW.employee_id; IF v_cnt = 0 THEN INSERT INTO employment &
31、#160; (employee_id, maxsalary) VALUES (:NEW.employee_id, :NEW.maxsalary); END IF; -检查是否存在该员工的工资信息 SEL
32、ECT COUNT(*) INTO v_cnt FROM salary WHERE employee_id = :NEW.employee_id AND MONTH = :NEW.MONTH;
33、160; IF v_cnt = 0 THEN INSERT INTO salary (employee_id, MONTH, amount) VALUES
34、0; (:NEW.employee_id, :NEW.MONTH, :NEW.amount); END IF;END employee_salary_rii;该触发器被建立后,执行上述insert操作,系统就会提示成功插入一条记录。但需要注意的是,这里的“成功插入一条记录”,只是Oracle并未发现触发器中有异常抛出,而根据insert语句中涉及的记录数作出一个判断。若触发器的主体什么都没有,只是一个空语句,Oracle也会报“成功插入一条记录”
35、。同样道理,即使在触发器主体里往多个表中插入十条记录,Oracle的返回也是“成功插入一条记录”。行级触发器可以解决大部分的问题,但是如果需要对本表进行扫描检查,比如要检查总的工资是否超限了,用行级触发器是不行的,因为行级触发器主体中不能有涉及到关联表的事务,这时就需要用到语句级触发器。以下是语句级触发器的语法:CREATE OR REPLACE TRIGGER trigger_name<before | after | instead of ><insert | update | delete > ON table_nameDECLAREBEGIN
36、 -触发器主体END;从语法定义上来看,行级触发器少了for each row,也不能使用when子句来限定入口条件,其他部分都是一样的,包括insert, update, delete和instead of都可以使用。7,例四:语句级触发器之一CREATE OR REPLACE TRIGGER salary_saiuAFTER INSERT OR UPDATE OF amount ON salaryDECLARE
37、 v_sumsalary NUMBER;BEGIN SELECT SUM(amount) INTO v_sumsalary FROM salary; IF v_sumsalary > 500000 THEN raise_application_error(-20001, '总工资超过500000');
38、 END IF;END;以上代码定义了一个语句级触发器,该触发器检查在insert和update了amount字段后操作后,工资表中所有工资记录累加起来是否超过500000,如果超过则抛出异常。从这个例子可以看出,语句级触发器可以对关联表表进行扫描,扫描得到的结果可以用来作为判断一致性的标志。需要注意的是,在before语句触发器主体和after语句触发器主体中对关联表进行扫描,结果是不一样的。在before语句触发器主体中扫描,扫描结果将不包括新插入和更新的记录,也就是说当以上代码换成before触发器后,以下语句将不报错:INSERT
39、 INTO salary(employee_id, month, amount) VALUEs(2, '200601', 600000)这是因为在主体中得到的v_sumsalary并不包括新插入的600000工资。另外,在语句级触发器中不能使用:new和:old对象,这一点和行级触发器是显著不同的。如果需要检查插入或更新后的记录,可以采用临时表技术。临时表是一种Oracle数据库对象,其特点是当创建数据的进程结束后,进程所创建的数据也随之清除。进程与进程不可以互相访问同一临时表中对方的数据,而且对临时表进行操作也不产生
40、undo日志,减少了数据库的消耗。具体有关临时表的知识,可以参看有关书籍。为了在语句级触发器中访问新插入后修改后的记录,可以增加行级触发器,将更新的记录插入临时表中,然后在语句级触发器中扫描临时表,获得修改后的记录。临时表的表结构一般与关联表的结构一致。8,例五:语句级触发器之二目的:限制每个员工的总工资不能超过50000,否则停止对该表操作。创建临时表create global temporary table SALARY_TMP( EMPLOYEE_ID NUMBER, MONTH
41、160; VARCHAR2(6), AMOUNT NUMBER)on commit delete rows;为了把操作记录插入到临时表中,创建行级触发器:CREATE OR REPLACE TRIGGER salary_raiuAFTER INSERT OR UPDATE OF amount ON salaryFOR EACH
42、0;ROWBEGIN INSERT INTO salary_tmp(employee_id, month, amount) VALUES(:NEW.employee_id, :NEW.MONTH, :NEW.amount);END;该触发器的作用是把更新后的记录信息插入到临时表中,如果更新了多条记录,则每条记录都会保存在临时表中。创建语句级触发器:CREATE OR REPLACE TRIGGER salary_saiAFTER INSERT OR UPDATE O
43、F amount ON salaryDECLARE v_sumsalary NUMBER;BEGIN FOR cur IN (SELECT * FROM salary_tmp) LOOP SELECT SUM(amount)
44、0; INTO v_sumsalary FROM salary WHERE employee_id = cur.employee_id; IF v_sumsalary &
45、gt; 50000 THEN raise_application_error(-20002, '员工累计工资超过50000'); END IF; DELETE FROM salary_tmp; END
46、60;LOOP;END;该触发器首先用游标从salary_tmp临时表中逐条读取更新或插入的记录,取employee_id,在关联表salary中查找所有相同员工的工资记录,并求和。若某员工工资总和超过50000,则抛出异常。如果检查通过,则清空临时表,避免下次检查相同的记录。执行以下语句:INSERT INTO salary(employee_id, month, amount) VALUEs(7, '200601', 20000);INSERT INTO salary(employee_id,&
47、#160;month, amount) VALUEs(7, '200602', 20000);INSERT INTO salary(employee_id, month, amount) VALUEs(7, '200603', 20000);在执行第三句时系统报错:ORA-20002:员工累计工资超过50000查询salary表,发现前两条记录正常插入了,第三条记录没有插入。如果系统结构比较复杂,而且触发器的代码比较多,在触发器主体中写过多的代码,对于维护来说是一个困难。这
48、时可以将所有触发器的代码写到同一个包中,不同的触发器代码以不同的存储过程封装,然后触发器主体中调用这部分代码。9,例六:用包封装触发器代码目的:改写例五,封装触发器主体代码创建代码包:CREATE OR REPLACE PACKAGE BODY salary_trigger_pck IS PROCEDURE load_salary_tmp(i_employee_id IN NUMBER, &
49、#160; i_month IN VARCHAR2,
50、 i_amount IN NUMBER) IS BEGIN INSERT INTO salary_tmp VALUES (i_employee_id, i_month, i_amount);
51、0; END load_salary_tmp; PROCEDURE check_salary IS v_sumsalary NUMBER; BEGIN FOR cur IN (SELECT * FROM salary_t
52、mp) LOOP SELECT SUM(amount) INTO v_sumsalary FROM salary WHERE emp
温馨提示
- 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
- 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
- 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
- 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
- 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
- 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
- 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
最新文档
- 2026年医院古医疗杂志模型馆共建合同
- 2025年数字营销服务平台开发项目可行性研究报告
- 2025年都市休闲农业发展项目可行性研究报告
- 2025年人工智能在金融机构中的应用可行性研究报告
- 纹身培训协议合同
- 生猪运输合同范本
- 主人保姆协议书
- 光伏补贴协议书
- 2025年共享电单车运营平台可行性研究报告
- 国家事业单位招聘2024中国科学院植物研究所特别研究助理(博士后)招聘笔试历年参考题库典型考点附带答案详解(3卷合一)
- 2025年淮北市相山区公开招考村(社区)后备干部66名笔试考试参考试题及答案解析
- 2025年贵州锦麟化工有限责任公司招聘备考题库及一套参考答案详解
- 2025年石家庄市公安局鹿泉分局公开招聘留置看护警务辅助人员30人的备考题库有答案详解
- 【数 学】2025-2026学年北师大版七年级数学上册期末综合提升卷III
- 车辆运营托管协议书
- 2025年云南省人民检察院聘用制书记员招聘(22人)笔试考试备考试题及答案解析
- 2024年电大法学专科行政法与行政诉讼法网考题库
- 起重机改造合同范本
- 2025中国医药健康产业股份有限公司总部有关领导人员选聘笔试历年参考题库附带答案详解
- 2026年企业财务共享中心建设方案
- 2025年科学方法与论文写作考试题及答案
评论
0/150
提交评论