oracle觸發(fā)器使用總結(jié)

觸發(fā)器 是特定事件出現(xiàn)的時候,自動執(zhí)行的代碼塊。類似于存儲過程,但是用戶不能直接調(diào)用他們。觸發(fā)器是許多關(guān)系數(shù)據(jù)庫系統(tǒng)都提供的一項技術(shù)。在ORACLE系統(tǒng)里,觸發(fā)器類似過程和函數(shù),都有聲明,執(zhí)行和異常處理過程的PL/SQL塊。
1.說明
1)觸發(fā)器是一種特殊的存儲過程,觸發(fā)器一般由事件觸發(fā)并且不能接受參數(shù),存儲器由語句塊去調(diào)用
2)觸發(fā)器分類:
1.DML觸發(fā)器: 創(chuàng)建在表上,由DML事件引發(fā)
2.instead of觸發(fā)器: 創(chuàng)建在視圖上并且只能在行級上觸發(fā),用于替代insert,delete等操作(由于oracle中不能直接對有兩個以上的表建立的視圖進(jìn)行DML操作,所以給出替代觸發(fā)器,它是專門為進(jìn)行視圖操作的一種處理方法)
3.DDL觸發(fā)器: 觸發(fā)事件時數(shù)據(jù)庫對象的創(chuàng)建和修改
4.數(shù)據(jù)庫事件觸發(fā)器:定義在數(shù)據(jù)庫或者模式上,由數(shù)據(jù)庫事件觸發(fā)
3)組成:
1.觸發(fā)事件:引發(fā)觸發(fā)器被觸發(fā)的事件 DML語句(INSERT, UPDATE, DELETE語句對表或視圖執(zhí)行數(shù)據(jù)處理操作)、DDL語句(如CREATE、ALTER、DROP語句在數(shù)據(jù)庫中創(chuàng)建、修改、刪除模式對象)、數(shù)據(jù)庫系統(tǒng)事件
?。ㄈ缦到y(tǒng)啟動或退出、異常錯誤)、用戶事件(如登錄或退出數(shù)據(jù)庫)。
2.觸發(fā)時間:即該觸發(fā)器是在觸發(fā)事件發(fā)生之前(BEFORE)還是之后(AFTER)觸發(fā)
3.觸發(fā)操作:觸發(fā)器觸發(fā)后要完成的事情
4.觸發(fā)對象:包括表、視圖、模式、數(shù)據(jù)庫。只有在這些對象上發(fā)生了符合觸發(fā)條件的觸發(fā)事件,觸發(fā)器才會執(zhí)行觸發(fā)操作。
5.觸發(fā)條件:由WHEN子句指定一個邏輯表達(dá)式。只有當(dāng)該表達(dá)式的值為TRUE時,遇到觸發(fā)事件才會自動執(zhí)行觸發(fā)操作。
6.觸發(fā)頻率:說明觸發(fā)器內(nèi)定義的動作被執(zhí)行的次數(shù)。即語句級(STATEMENT)觸發(fā)器和行級(ROW)觸發(fā)器。(比如delete多條數(shù)據(jù)時,行級觸發(fā)器可能會執(zhí)行多次,語句級觸發(fā)器只會觸發(fā)一次)
2.語法
1)說明
不同類型的觸發(fā)器例如DML觸發(fā)器,Instead of觸發(fā)器,系統(tǒng)觸發(fā)器語法格式區(qū)別較大
2)一般語法
CREATE [OR REPLACE] TIGGER觸發(fā)器名 觸發(fā)時間 觸發(fā)事件 ON表名/視圖名 [FOR EACH ROW] //加上FOR EACH ROW 即為行級觸發(fā)器,不加時為語句級觸發(fā)器 BEGIN pl/sql語句 END
create [or replace] trigger [schema.]trigger_name {before | after | instead of} {delete [or insert][or update [of column,...n]]} on [schema.]table_name | view_name [for each row [when(condition)]] sql_statement[,...n]
例如:
CREATE OR REPLACE TRIGGER trigger_name < before | after | instead of > < insert | update | delete> ON table_name [FOR EACH ROW] WHEN (condition) DECLARE BEGIN END;
3)instead of 觸發(fā)器語法
語法:
CREATE [OR REPLACE] TRIGGER trigger_name INSTEAD OF{INSERT|DELETE|UPDATE [OF COLUMN...]}[OR {INSERT| DELETE| UPDATE [OF COLUMN...]}]ON VIEW_NAME[REFFERENCING{OLD [AS] OLD | NEW [AS] NEW| PARENT AS PARENT}] // 可以指定相關(guān)名稱,當(dāng)前的默認(rèn)相關(guān)名稱為OLD和NEW,應(yīng)用相關(guān)名稱時需要加:[FOR EACH ROW] //instead of 觸發(fā)器只能在行級上觸發(fā),因為沒有必要指定[WHEN CONDITION]DECLAREBEGINEND;
說明:INSTEAD OF 用于對視圖的DML觸發(fā),由于視圖可能有多個表進(jìn)行聯(lián)結(jié)而成,因而并非所有的聯(lián)結(jié)均可更新,運用 INSTEAD OF 觸發(fā)器可完成相應(yīng)的操作。
3.實例
創(chuàng)建測試表格:
CREATE TABLE "HNZC"."TRIGGERTEST" ( "ID" VARCHAR2(20 BYTE), "NAME" VARCHAR2(20 BYTE), "SCORE" NUMBER );create table tab1 select * from triggertest;
1)DML觸發(fā)器/行級觸發(fā)器
觸發(fā)器如下:
CREATE OR REPLACE TRIGGER TRIGGER1 AFTER INSERT ON TRIGGERTEST //插入后觸發(fā)FOR EACH ROW //行級觸發(fā)器BEGIN INSERT INTO tab1(ID,NAME) VALUES('22','33');END;
執(zhí)行語句:
insert into triggertest (id) values ('aabbcc');
語句執(zhí)行結(jié)束,表tab1中新增加一條數(shù)據(jù)
2)限制對表的修改(例如非工作時間不能修改某些表)
觸發(fā)器如下:
CREATE OR REPLACE TRIGGER TRIGGER1 AFTER INSERT ON TRIGGERTEST FOR EACH ROW BEGIN IF(TO_CHAR(SYSDATE,'DAY') IN ('星期三','星期天')) THEN RAISE_APPLICATION_ERROR(-20001,'不是上班時間,不能修改表格triggertest'); END IF;END;
執(zhí)行語句:
insert into triggertest (id) values ('aabbcc');
今天周三因而輸出結(jié)果為:
在行 1 上開始執(zhí)行命令時出錯:insert into triggertest (id) values ('aabbcc')錯誤報告:SQL 錯誤: ORA-20001: 不是上班時間,不能修改表格triggertestORA-06512: 在 "HNZC.TRIGGER1", line 3ORA-04088: 觸發(fā)器 'HNZC.TRIGGER1' 執(zhí)行過程中出錯
通常對表的修改限制如下(即周一至周五9——18點能修改表格)
CREATE OR REPLACE TRIGGER TRIGGER1 BEFORE INSERT OR DELETE OR UPDATE ON TRIGGERTEST FOR EACH ROW BEGIN IF(TO_CHAR(SYSDATE,'DAY') IN ('星期六','星期天')) OR(TO_CHAR(SYSDATE,'HH24:MI') NOT BETWEEN '9:00' AND '18:00') THEN RAISE_APPLICATION_ERROR(-20001,'不是上班時間,不能修改表格triggertest'); END IF;END;
3)增加限制條件(如不能更改某個員工的記錄)
觸發(fā)器如下:(如下實現(xiàn)月兒的分?jǐn)?shù)只能增加)
CREATE OR REPLACE TRIGGER TRIGGER1 BEFORE INSERT OR DELETE OR UPDATE ON TRIGGERTEST FOR EACH ROW WHEN(OLD.NAME='月兒')BEGIN CASE WHEN UPDATING('SCORE') THEN IF:NEW.SCORE<:OLD.SCORE THEN RAISE_APPLICATION_ERROR(-20001,'月兒的分?jǐn)?shù)只能提升不能下降'); END IF; END CASE;END;
當(dāng)前月兒的分?jǐn)?shù)為20
當(dāng)修改為10時出錯
UPDATE "HNZC"."TRIGGERTEST" SET SCORE = '10' WHERE ROWID = 'AAAdEzAAPAAAAH+AAB' AND ORA_ROWSCN = '47685303'ORA-20001: 月兒的分?jǐn)?shù)只能提升不能下降ORA-06512: 在 "HNZC.TRIGGER1", line 4ORA-04088: 觸發(fā)器 'HNZC.TRIGGER1' 執(zhí)行過程中出錯
當(dāng)修改為30時成功
UPDATE "HNZC"."TRIGGERTEST" SET SCORE = '30' WHERE ROWID = 'AAAdEzAAPAAAAH+AAB' AND ORA_ROWSCN = '47685303'提交成功
4)在觸發(fā)器中調(diào)用存儲過程
觸發(fā)器為:
CREATE OR REPLACE TRIGGER TRIGGER1 BEFORE INSERT OR DELETE OR UPDATE ON TRIGGERTEST FOR EACH ROW BEGIN TESTPRO1();END;
存儲過程為:
create or replacePROCEDURE TESTPRO1 AS BEGIN insert into tab1(id,name,score) VALUES('AAA','BBB',200);END TESTPRO1;
執(zhí)行完畢后tab1中增加一條數(shù)據(jù)
5)級聯(lián)更新
觸發(fā)器如下(triggertest表中name修改時同時修改tab1中的name)
create or replacePROCEDURE TESTPRO1 AS BEGIN insert into tab1(id,name,score) VALUES('AAA','BBB',200);END TESTPRO1;
執(zhí)行語句:
update triggertest set name= '水兒' where name='月兒';
結(jié)果:tab1中name為月兒的也更改為水兒
6)instead of觸發(fā)器
TABLE STUDENT表格數(shù)據(jù)如下
創(chuàng)建視圖student_view
CREATE OR REPLACE VIEW STUDNET_VIEW AS SELECT CLASSID,AVG(SCORE) AVERAGE_SCORE FROM STUDENTGROUP BY CLASSID;
視圖數(shù)據(jù)如下:
對視圖student_view 執(zhí)行如下操作:
DELETE FROM STUDNET_VIEW WHERE CLASSID='111';
執(zhí)行結(jié)果:
錯誤報告:SQL 錯誤: ORA-01732: 此視圖的數(shù)據(jù)操縱操作非法01732. 00000 - "data manipulation operation not legal on this view"
解決方法:創(chuàng)建INSTEAD OF 視圖
CREATE OR REPLACE TRIGGER STUDENT_VIEW_DELETE INSTEAD OF DELETE ON STUDNET_VIEW FOR EACH ROWBEGIN DELETE FROM STUDENT WHERE CLASSID=:OLD.CLASSID;END STUDENT_VIEW_DELETE;
執(zhí)行刪除語句
DELETE FROM STUDNET_VIEW WHERE CLASSID='111';
執(zhí)行結(jié)果:刪除成功
1 行已刪除。
4.注意事項
1) 在觸發(fā)器的執(zhí)行部分只能用DML語句(SELECT、INSERT、UPDATE、DELETE),不能使用DDL語句(CREATE、ALTER、DROP)
2) 觸發(fā)器中不能使用commit語句,觸發(fā)器的操作與觸發(fā)事件(INSERT,UPDATE,DELETE)一起進(jìn)行COMMIT和ROLLBACK;
3) 一個表上的觸發(fā)器越多,對于表的DML操作性能影響越大
4) 觸發(fā)器最大為32K
相關(guān)文章
Oracle VM VirtualBox虛擬機的安裝使用圖文教程
相信很多朋友都知道Oracle VM VirtualBox。我也不例外,一次無意之間交流測試環(huán)境的時候朋友提出了他一直使用Oracle VM VirtualBox,好奇之下便安裝使用了2014-05-19oracle導(dǎo)入導(dǎo)出表的詳細(xì)圖解
這篇文章主要介紹了oracle導(dǎo)入導(dǎo)出表的詳細(xì)圖解,需要的朋友可以參考下2014-04-29使用oracle的SQL Developer創(chuàng)建用戶的方法
這篇文章主要介紹了使用oracle的SQL Developer創(chuàng)建用戶的方法,需要的朋友可以參考下2014-04-29使用數(shù)據(jù)庫客戶端工具Oracle SQL Developer加載第三方驅(qū)動連接mysql的
Oracle SQL Developer是Oracle官方提供的數(shù)據(jù)庫連接工具。不僅可以連接自己的數(shù)據(jù)庫(Oracle),而且還可以連接多種其他的數(shù)據(jù)庫,下面以mysql為例介紹一下使用方法2014-04-25Oracle VM VirtualBox工具怎么用 VirtualBox虛擬機安裝教程圖解
有很多朋友對虛擬機很陌生,沒用過虛擬機的童鞋們覺得虛擬機的安裝很難!其實虛擬機的安裝很簡單,安裝虛擬機的方法有很多種,本文為大家介紹Oracle VM VirtualBox虛擬機安2014-01-07sql server 2005如何導(dǎo)出數(shù)據(jù)到oracle 11g
sql server 2005如何導(dǎo)出數(shù)據(jù)到oracle 11g ,需要的朋友可以看看,導(dǎo)出的方法很簡單2013-08-28使用Navicat for Oracle工具連接oracle的圖文教程
今天上網(wǎng)的時候偶然發(fā)現(xiàn)了一款oracle的客戶端的圖形化管理和開發(fā)工具,當(dāng)看到這個界面的時候,感覺很舒服,便上網(wǎng)搜了一下這個工具,看百度百科之后感覺很出乎我的意料,這2013-05-24Navicat Premium 10.0.5中文版 oracle連接字符集報錯解決辦法
針對Navicat Premium 10.X 中文版解決方法,需要的朋友可以參考下2013-05-24oracle 9i服務(wù)器端中文安裝教程圖解 32位/64位
oracle9i是業(yè)界第一個完整、簡單的用于互聯(lián)網(wǎng)的新一代智能化的、協(xié)作各種應(yīng)用的軟件基礎(chǔ)架構(gòu),是Oracle9iDatabase, Oracle 9i Application Server 和Oracle9i Developer Su2017-07-28Oracle Database 12c數(shù)據(jù)庫中文配置安裝圖解教程(詳細(xì)安裝步驟)
Oracle Database(Oracle數(shù)據(jù)庫)是一套甲骨文公司以高級結(jié)構(gòu)化查詢語言(SQL)為基礎(chǔ)設(shè)計的大型關(guān)系數(shù)據(jù)庫,能在對稱多CPU的系統(tǒng)上提供并行處理,下面整理了Oracle Database 12017-07-28