首頁 資料庫 mysql教程 oracle触发器实例

oracle触发器实例

Jun 07, 2016 pm 03:24 PM
oracle 什麼 儲存 實例 資料庫 觸發器

一、 什么是触发器? 数据库触发器是一个存储的PL/SQL程序块,它与一个基表联系,当在表上执行特定的数据库维护(插入、删除、更新这三种操作)时,隐含地执行一个PL/SQL程序块。 二、触发器的作用: 。防止非法的数据库操纵、维护数据库安全 。对数据库的操

一、 什么是触发器?
         数据库触发器是一个存储的PL/SQL程序块,它与一个基表联系,当在表上执行特定的数据库维护(插入、删除、更新这三种操作)时,隐含地执行一个PL/SQL程序块。

二、触发器的作用:
              。防止非法的数据库操纵、维护数据库安全
              。对数据库的操作进行审计,存储历史数据
              。完成数据库初始化处理
              。控制数据库的数据完整性
              。进行相关数据的修改
              。完成数据复制
              。自动完成数据库统计计算
              。限制数据库操作的时间、权限等,控制实体的安全性

 
 三、触发器的组成:
            1、触发时间:触发器事件的时间次序(before,afer)[2]

            2、触发事件:什么SQL语句会引起触发器触发(Insert, delete, update)[3]

            3、触发子体:触发器触发时要执行的操作(一个完整的PL/SQL程序)

            4、触发类型:触发器被执行的次数(语句级、行级)[2] //语句级只执行一次,行级会执行多次。

                 [*]一个表上最多可以创建12个不同类型的触发器:3*2*2 = 12

 

 四、创建触发器注意事项:
             1、在触发器中可以调用存储过程、包;在存储过程中不得调用触发器。

             2、在触发器中不得使用commit, rollback,savepoint语句。

             3、在触发器中不得间接调用含有commit,rollback, savepoint的语句的存储过程及函数。

 

 五、创建语句级触发器:
                   语句级触发器: 请参考PowerPoint教程:存储过程1.ppt[Page19] 该触发器在数据库操作时只执行一次。
                说明:
                      。update中的of是可选项,用于指定语句要修改的列
                      。要创建的触发器已经存在时,使用replace选项

1.   //例1:before型触发器:   

2.   Create or replace trigger DelEmp   

3.     before delete on emp   

4.     Begin  

5.      if (To_Char(sysdate,'dy') in ('星期六','星期日') or  

6.           To_number(To_Char(sysdate,'hh24'))  not between  8   and 18)   

7.      then dbms_output.put_line('现在是非工作时间,请退出!!!');   

8.      end if;   

9.     End;  

[触发器数据字典]
SQL> select table_owner, table_name,trigger_body from user_triggers wheretrigger_name='DELEMP';

1.  //例2:After型触发器:   

2.  Create or replace trigger InsertEmp  

3.  after insert on emp     // 如果是before,就会比after的结果少一名。   

4.  Declare  

5.   v_empcount number(7);   

6.  Begin  

7.   select count(*) into v_empcount from emp;  

8.   dbms_output.put_line('目前员工总数已达到:'|| v_empcount|| '名。');   

9.  End;  

 

1.  //例3:多个触发条件   

2.  Create or replace trigger ChangeEmp  

3.  before delete or insert or update on emp  

4.  Begin  

5.     if (To_Char(sysdate,'dy') in ('星期六','星期日') or  

6.          To_number(To_Char(sysdate,'hh24'))  not between 8 and 18)  

7.     then dbms_output.put_line('现在是非工作时间,请不要修改数据!!!');  

8.     end if;  

9.  End;   

10.   

11. // 更完善的写法:   

12. Create or replace trigger ChangeEmp  

13.   before delete or insert or update  on emp  

14.   Begin  

15.    if (DELETING and (To_Char(sysdate,'dy') in ('星期六','星期日') or  

16.         To_number(To_Char(sysdate,'hh24'))  not between  8   and 18))  

17.    then dbms_output.put_line('现在是非工作时间,不要删除数据!');  

18.   

19.    elsif (UPDATING and (To_Char(sysdate,'dy') in ('星期六','星期日') or  

20.         To_number(To_Char(sysdate,'hh24'))  not between  8 and 18))  

21.    then dbms_output.put_line('现在是非工作时间,不要更新数据!');  

22.   

23.    elsif (INSERTING and (To_Char(sysdate,'dy') in ('星期六','星期日') or  

24.         To_number(To_Char(sysdate,'hh24'))  not between  8   and 18))  

25.    then dbms_output.put_line('现在是非工作时间,不要插入数据!');  

26.   

27.    end if;  

28.   End;   

 

        六、创建行级触发器:
           等级触发器:增加选项for each row, 使触发器在每一行上触发。

            1、创建行级触发器注意事项:
                     (1) 在行级触发器中,在列名前增加old表示该列修改前值,增加new表示该列修改后值。
                     (2) 在PL/SQL中引用时,前边增加冒号。
 

1.  [例4: 行级触发器] //必须是对所有的行进行操作才行。   

2.  Create or Replace trigger UpdateEmp  

3.  Before update on emp   

4.  for each row   

5.  Begin  

6.   dbms_output.put_line(:old.sal||'--------->'||:new.sal);  

7.  End;  

 

1.  [例5:保存历史数据,这种使用方法很重要,用来保存关键表的历史数据]   

2.  CReate or Replace trigger ChangeEmp  

3.  Before update or delete on emp  

4.  for each row  

5.  Begin  

6.   Insert into oldemp(empno, ename,job,hiredate,sal)  

7.   values(:old.empno,:old.ename,:old.job,sysdate,:old.sal);  

8.  End;  

9.    

10. SQL> create table oldemp  

11.  as select empno, ename,job,hiredate,sal from emp where 1>2;  

1.  [例6:修改外键]   

2.  Create or Replace trigger UpdateDept  

3.  after update on dept  

4.  for each row  

5.  Begin  

6.   update emp   

7.    set emp.deptno = :new.deptno  

8.    where emp.deptno = :old.deptno;  

9.  End;  

1.  [例7:删除外键、删除相关数据]   

2.  Create or Replace trigger DeleteDept  

3.  before delete on dept  

4.  for each row  

5.  Begin  

6.   delete from emp where deptno = :old.empno;  

7.  End;  

 

        七、触发器管理
            1、使触发器失效:

1.  SQL> alter trigger 触发器名称 disable;  // 失效  

2.  SQL> Alter Trigger 触发器名称 enable;  // 生效  

3.    

4.  SQL> Alter table 表名 DISABLE all triggers; // 一个表上的所有触发器失效  

5.  SQL> Alter table 表名 ENABLE all triggers; // 使一个表上的所有触发器生效  

6.    

7.  SQL> Drop Trigger 触发器名;  // 删除触发器;  

 

 

 

 

 

 

--test表执行DML语句时,将相关信息记录到日志表
--创建测试表
CREATE TABLE test ( t_id NUMBER(4), t_name VARCHAR2(20), t_age NUMBER(2), t_sexCHAR );
--创建记录测试表
CREATE TABLE test_log ( l_user VARCHAR2(15), l_type VARCHAR2(15), l_dateVARCHAR2(30) );
--创建触发器
CREATE OR REPLACE TRIGGER test_trigger
AFTER DELETE OR INSERT OR UPDATE ON test
DECLARE v_type test_log.l_type%TYPE;
BEGIN
IF INSERTING THEN
--INSERT触发
v_type := 'INSERT';
DBMS_OUTPUT.PUT_LINE('记录已经成功插入,并已记录到日志');
ELSIF UPDATING THEN
--UPDATE触发
v_type := 'UPDATE';
DBMS_OUTPUT.PUT_LINE('记录已经成功更新,并已记录到日志');
ELSIF DELETING THEN v_type := 'DELETE';
DBMS_OUTPUT.PUT_LINE('记录已经成功删除,并已记录到日志');
END IF;
INSERT INTO test_log VALUES(user,v_type, TO_CHAR(sysdate,'yyyy-mm-ddhh24:mi:ss'));
END;
/
--下面我们来分别执行DML语句
INSERT INTO test VALUES(101,'zhao',22,'M');
UPDATE test SET t_age = 30 WHERE t_id = 101;
DELETE test WHERE t_id = 101;
--然后查看效果
SELECT * FROM test;
SELECT * FROM test_log;
--实例2------------------------
--创建触发器,它将映射emp表中每个部门的总人数和总工资
--创建映射表
CREATE TABLE dept_sal AS
SELECT deptno,COUNT(empno) AS total_emp,SUM(sal) AS total_sal FROM emp GROUP BYdeptno;
DESC dept_sal;
--创建触发器
CREATE OR REPLACE TRIGGER emp_info
AFTER INSERT OR UPDATE OR DELETE ON emp
DECLARE CURSOR cur_emp IS
SELECT deptno,COUNT(empno) AS total_emp,SUM(sal) AS total_sal FROM emp GROUP BYdeptno;
BEGIN DELETE dept_sal;
--触发时首先删除映射表信息
FOR v_emp IN cur_emp LOOP
--DBMS_OUTPUT.PUT_LINE(v_emp.deptno || v_emp.total_emp || v_emp.total_sal);
--插入数据
INSERT INTO dept_sal VALUES(v_emp.deptno,v_emp.total_emp,v_emp.total_sal);
END LOOP;
END;
/
--对emp表进行DML操作
INSERT INTO emp(empno,deptno,sal) VALUES('123','10',10000);
SELECT * FROM dept_sal;
DELETE EMP WHERE empno=123;
SELECT * FROM dept_sal;
--实例3------------------------
--创建触发器,它记录表的删除数据
--创建表
CREATE TABLE employee ( id VARCHAR2(4) NOT NULL, name VARCHAR2(15) NOT NULL,age NUMBER(2) NOT NULL, sex CHAR NOT NULL ); DESC employee;
--插入数据
INSERT INTO employee VALUES('e101','zhao',23,'M');
INSERT INTO employee VALUES('e102','jian',21,'F');
--创建记录表
CREATE TABLE old_employee AS SELECT * FROM employee;
DESC old_employee;
--创建触发器
CREATE OR REPLACE TRIGGER tig_old_emp
AFTER DELETE ON employee
FOR EACH ROW
--语句级触发,即每一行触发一次
BEGIN
INSERT INTO old_employee
VALUES(:old.id,:old.name,:old.age,:old.sex);
--:old代表旧值
END;
/ --下面进行测试
DELETE employee;
SELECT * FROM old_employee;
--实例4------------------------
--创建触发器,利用视图插入数据 --创建表
CREATE TABLE tab1 (tid NUMBER(4) PRIMARY KEY,tname VARCHAR2(20),tageNUMBER(2));
CREATE TABLE tab2 (tid NUMBER(4),ttel VARCHAR2(15),tadr VARCHAR2(30));
--插入数据
INSERT INTO tab1 VALUES(101,'zhao',22);
INSERT INTO tab1 VALUES(102,'yang',20);
INSERT INTO tab2 VALUES(101,'13761512841','AnHuiSuZhou');
INSERT INTO tab2 VALUES(102,'13563258514','AnHuiSuZhou');
--创建视图连接两张表
CREATE VIEW tab_view AS
SELECT tab1.tid,tname,ttel,tadr FROM tab1,tab2 WHERE tab1.tid = tab2.tid; --创建触发器
CREATE OR REPLACE TRIGGER tab_trigger
INSTEAD OF INSERT ON tab_view
BEGIN
INSERT INTO tab1(tid,tname) VALUES(:new.tid,:new.tname);
INSERT INTO tab2(ttel,tadr) VALUES(:new.ttel,:new.tadr);
END;
/
--现在就可以利用视图插入数据
INSERT INTO tab_view VALUES(105,'zhaoyang','13886681288','beijing');
--查看效果
SELECT * FROM tab_view;
--实例5------------------------
--创建触发器,比较emp表中更新的工资
CREATE OR REPLACE TRIGGER sal_emp BEFORE UPDATE ON emp FOR EACH ROW BEGIN IF:OLD.sal > :NEW.sal THEN DBMS_OUTPUT.PUT_LINE('工资减少');
ELSIF :OLD.sal ELSE DBMS_OUTPUT.PUT_LINE('工资未作任何变动');
END IF;
DBMS_OUTPUT.PUT_LINE('更新前工资 :' || :OLD.sal);
DBMS_OUTPUT.PUT_LINE('更新后工资 :' || :NEW.sal);
END;
/ --执行UPDATE查看效果
UPDATE emp SET sal = 3000 WHERE empno = '7788';
--实例6------------------------
--创建触发器,将操作CREATE、DROP存储在log_info表
--创建表
CREATE TABLE log_info ( manager_user VARCHAR2(15), manager_date VARCHAR2(15),manager_type VARCHAR2(15), obj_name VARCHAR2(15), obj_type VARCHAR2(15) );
--创建触发器
CREATE OR REPLACE TRIGGER trig_log_info
AFTER CREATE OR DROP ON SCHEMA
BEGIN INSERT INTO log_info
VALUES(USER,SYSDATE,SYS.DICTIONARY_OBJ_NAME,SYS.DICTIONARY_OBJ_OWNER,SYS.DICTIONARY_OBJ_TYPE);
END;
/
--测试语句
CREATE TABLE a(id NUMBER);
CREATE TYPE aa AS OBJECT(id NUMBER);
/
DROP TABLE a;
DROP TYPE aa;
--查看效果
SELECT * FROM log_info;
--相关数据字典-----------------------------------------------------//
SELECT * FROM USER_TRIGGERS;
SELECT * FROM ALL_TRIGGERS;
SELECT * FROM DBA_TRIGGERS;
--必须以DBA身份登陆才能使用此数据字典
--启用和禁用
ALTER TRIGGER trigger_name DISABLE;
ALTER TRIGGER trigger_name ENABLE;

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

AI Hentai Generator

AI Hentai Generator

免費產生 AI 無盡。

熱門文章

R.E.P.O.能量晶體解釋及其做什麼(黃色晶體)
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.最佳圖形設置
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.如果您聽不到任何人,如何修復音頻
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
WWE 2K25:如何解鎖Myrise中的所有內容
3 週前 By 尊渡假赌尊渡假赌尊渡假赌

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

iOS 18 新增「已復原」相簿功能 可找回遺失或損壞的照片 iOS 18 新增「已復原」相簿功能 可找回遺失或損壞的照片 Jul 18, 2024 am 05:48 AM

蘋果公司最新發布的iOS18、iPadOS18以及macOSSequoia系統為Photos應用程式增添了一項重要功能,旨在幫助用戶輕鬆恢復因各種原因遺失或損壞的照片和影片。這項新功能在Photos應用的"工具"部分引入了一個名為"已恢復"的相冊,當用戶設備中存在未納入其照片庫的圖片或影片時,該相冊將自動顯示。 "已恢復"相簿的出現為因資料庫損壞、相機應用未正確保存至照片庫或第三方應用管理照片庫時照片和視頻丟失提供了解決方案。使用者只需簡單幾步

在PHP中使用MySQLi建立資料庫連線的詳盡教學 在PHP中使用MySQLi建立資料庫連線的詳盡教學 Jun 04, 2024 pm 01:42 PM

如何在PHP中使用MySQLi建立資料庫連線:包含MySQLi擴充(require_once)建立連線函數(functionconnect_to_db)呼叫連線函數($conn=connect_to_db())執行查詢($result=$conn->query())關閉連線( $conn->close())

如何在PHP中處理資料庫連線錯誤 如何在PHP中處理資料庫連線錯誤 Jun 05, 2024 pm 02:16 PM

PHP處理資料庫連線報錯,可以使用下列步驟:使用mysqli_connect_errno()取得錯誤代碼。使用mysqli_connect_error()取得錯誤訊息。透過擷取並記錄這些錯誤訊息,可以輕鬆識別並解決資料庫連接問題,確保應用程式的順暢運作。

如何在 Golang 中使用資料庫回呼函數? 如何在 Golang 中使用資料庫回呼函數? Jun 03, 2024 pm 02:20 PM

在Golang中使用資料庫回呼函數可以實現:在指定資料庫操作完成後執行自訂程式碼。透過單獨的函數新增自訂行為,無需編寫額外程式碼。回調函數可用於插入、更新、刪除和查詢操作。必須使用sql.Exec、sql.QueryRow或sql.Query函數才能使用回呼函數。

如何用 Golang 連接遠端資料庫? 如何用 Golang 連接遠端資料庫? Jun 01, 2024 pm 08:31 PM

透過Go標準庫database/sql包,可以連接到MySQL、PostgreSQL或SQLite等遠端資料庫:建立包含資料庫連接資訊的連接字串。使用sql.Open()函數開啟資料庫連線。執行SQL查詢和插入操作等資料庫操作。使用defer關閉資料庫連線以釋放資源。

如何使用C++處理資料庫連線和操作? 如何使用C++處理資料庫連線和操作? Jun 01, 2024 pm 07:24 PM

在C++中使用DataAccessObjects(DAO)函式庫連接和操作資料庫,包括建立資料庫連線、執行SQL查詢、插入新記錄和更新現有記錄。具體步驟為:1.包含必要的函式庫語句;2.開啟資料庫檔案;3.建立Recordset物件執行SQL查詢或操作資料;4.遍歷結果或依照特定需求更新記錄。

PHP與不同資料庫的連接:MySQL、PostgreSQL、Oracle和更多 PHP與不同資料庫的連接:MySQL、PostgreSQL、Oracle和更多 Jun 01, 2024 pm 03:02 PM

PHP連接資料庫指南:MySQL:安裝MySQLi擴展,建立連線(servername、username、password、dbname)。 PostgreSQL:安裝PgSQL擴展,建立連線(host、dbname、user、password)。 Oracle:安裝OracleOCI8擴展,建立連線(servername、username、password)。實戰案例:取得MySQL資料、PostgreSQL查詢、OracleOCI8更新記錄。

Go WebSocket 如何與資料庫整合? Go WebSocket 如何與資料庫整合? Jun 05, 2024 pm 03:18 PM

如何將GoWebSocket與資料庫整合:設定資料庫連線:使用database/sql包連接到資料庫。將WebSocket訊息儲存到資料庫:使用INSERT語句將訊息插入資料庫。從資料庫檢索WebSocket訊息:使用SELECT語句檢索資料庫中的消息。

See all articles