oracle中建立觸發器

從csdn上面看到一個如何建立觸發器的問題,感受本身頗有必要保存學習,特寫下來:
條件:
現有A、B兩張表
A: 工號 姓名 密碼 性別 年齡 。。。
B: 工號 姓名 密碼
當對A表中的「密碼」字段進行修改時,B表中的「密碼」字段一樣被修改,
建立觸發器:代碼以下:
------------------------------------------------
create or replace trigger trg_a
before update of 密碼 on A
for each row
begin
    update B set 密碼=:new.密碼 where 工號=:new.工號 and 姓名=:new.姓名;
end;
----------------------------------------
CREATE OR REPLACE TRIGGER tr_a
  AFTER UPDATE ON a
  FOR EACH ROW
BEGIN
  IF updating('密碼') THEN
    UPDATE b SET b.密碼 = :new.密碼 WHERE b.工號 = :new.工號;
  END IF;
END;
-----------------------------------------
CREATE OR REPLACE TRIGGER tr_a
  AFTER UPDATE of 密碼 ON a --創建在列上面的觸發器
  FOR EACH ROW
BEGIN
  UPDATE b SET b.密碼 = :new.密碼 WHERE b.工號 = :new.工號;
END;
這是三種不一樣的答案,可是都相似,因爲沒有裝oracle數據庫因此不能進行測試,從上面能夠看出怎樣建立列上面的觸發器,
這個雖然簡單可是不接觸時就感受很難,沒辦法下手。注意學習oracle中建立觸發器的語法。
下面爲從網上找的關於觸發器語法的學習總結一下:
觸發器
是特定事件出現的時候,自動執行的代碼塊。相似於存儲過程,可是用戶不能直接調用他們。
功能:
一、 容許/限制對錶的修改
二、 自動生成派生列,好比自增字段
三、 強制數據一致性
四、 提供審計和日誌記錄
五、 防止無效的事務處理
六、 啓用複雜的業務邏輯
開始
create trigger biufer_employees_department_id
 before insert or update
  of department_id
  on employees
 referencing old as old_value
     new as new_value
 for each row
 when (new_value.department_id<>80 )
begin
 :new_value.commission_pct :=0;
end;
/
觸發器的組成部分:
一、 觸發器名稱
二、 觸發語句
三、 觸發器限制
四、 觸發操做
一、 觸發器名稱
create trigger biufer_employees_department_id
命名習慣:
biufer(before insert update for each row)
employees 表名
department_id 列名
二、 觸發語句
好比:
表或視圖上的dml語句
ddl語句
數據庫關閉或啓動,startup shutdown 等等
before insert or update
  of department_id
  on employees
 referencing old as old_value
     new as new_value
 for each row
說明:
一、 不管是否規定了department_id ,對employees表進行insert的時候
二、 對employees表的department_id列進行update的時候
三、 觸發器限制
when (new_value.department_id<>80 )
限制不是必須的。此例表示若是列department_id不等於80的時候,觸發器就會執行。
其中的new_value是表明跟新以後的值。
四、 觸發操做
是觸發器的主體
begin
 :new_value.commission_pct :=0;
end;
主體很簡單,就是將更新後的commission_pct列置爲0
觸發:
insert into employees(employee_id,
last_name,first_name,hire_date,job_id,email,department_id,salary,commission_pct )
values( 12345,’chen’,’donny’, sysdate, 12, ‘donny@hotmail.com’,60,10000,.25);
select commission_pct from employees where employee_id=12345;
觸發器不會通知用戶,便改變了用戶的輸入值。

觸發器類型:
一、 語句觸發器
二、 行觸發器
三、 instead of 觸發器
四、 系統條件觸發器
五、 用戶事件觸發器
 
一、 語句觸發器
是在表上或者某些狀況下的視圖上執行的特定語句或者語句組上的觸發器。可以與insert、update、delete或者組合上進行關聯。可是不管使用什麼樣的組合,各個語句觸發器都只會針對指定語句激活一次。好比,不管update多少行,也只會調用一次update語句觸發器。
例子:
須要對在表上進行dml操做的用戶進行安全檢查,看是否具備合適的特權。
create table foo(a number);
create trigger biud_foo
 before insert or update or delete
  on foo
begin
 if user not in (‘donny’) then
  raise_application_error(-20001, ‘you don’t have access to modify this table.’);
 end if;
end;
/
即便sys,system用戶也不能修改foo表

對修改表的時間、人物進行日誌記錄。
一、 創建試驗表
create table employees_copy as select *from hr.employees
二、 創建日誌表
create table employees_log(
  who varchar2(30),
  when date);
三、 在employees_copy表上創建語句觸發器,在觸發器中填充employees_log 表。
create or replace trigger biud_employee_copy
  before insert or update or delete
   on employees_copy
 begin
  insert into employees_log(
   who,when)
  values( user, sysdate);
 
 end;
 /
四、 測試
update employees_copy set salary= salary*1.1;
select *from employess_log;
五、 肯定是哪一個語句起做用?
便是insert/update/delete中的哪個觸發了觸發器?
能夠在觸發器中使用inserting / updating / deleting 條件謂詞,做判斷:
begin
  if inserting then
   -----
  elsif updating then
   -----
  elsif deleting then
   ------
  end if;
end;
if updating(‘col1’) or updating(‘col2’) then
  ------
end if;

一、 修改日誌表
alter table employees_log
  add (action varchar2(20));
二、 修改觸發器,以便記錄語句類型。
create or replace trigger biud_employee_copy
  before insert or update or delete
   on employees_copy
 declare
  l_action employees_log.action%type;
 begin
  if inserting then
   l_action:=’insert’;
  elsif updating then
   l_action:=’update’;
  elsif deleting then
   l_action:=’delete’;
  else
   raise_application_error(-20001,’you should never ever get this error.’);
  insert into employees_log(
   who,action,when)
  values( user, l_action,sysdate);
 end;
 /
三、 測試
insert into employees_copy( employee_id, last_name, email, hire_date, job_id)
 values(12345,’chen’,’donny@hotmail’,sysdate,12);
select *from employees_log
update employees_copy set salary=50000 where employee_id = 12345;
二、 行觸發器
是指爲受到影響的各個行激活的觸發器,定義與語句觸發器相似,有如下兩個例外:
一、 定義語句中包含for each row子句
二、 在before……for each row觸發器中,用戶能夠引用受到影響的行值。
好比:
定義:
create trigger biufer_employees_department_id
 before insert or update
  of department_id
  on employees_copy
 referencing old as old_value
     new as new_value
 for each row
 when (new_value.department_id<>80 )
begin
 :new_value.commission_pct :=0;
end;
/
referencing 子句:
執行dml語句以前的值的默認名稱是 :old ,以後的值是 :new
insert 操做只有:new
delete 操做只有 :old
update 操做二者都有
referencing子句只是將new 和old重命名爲new_value和old_value,目的是避免混淆。好比操做一個名爲new的表時。
做用不很大。
 
:爲主健生成自增序列號
drop table foo;
create table foo(id number, data varchar2(20));
create sequence foo_seq;
create or replace trigger bifer_foo_id_pk
 before insert on foo
 for each row
begin
 select foo_seq.nextval into :new.id from dual;
end;
/
insert into foo(data) values(‘donny’);
insert into foo values(5,’chen’);
select * from foo;
三、 instead of 觸發器更新視圖
create or replace view company_phone_book as
 select first_name||’, ’||last_name name, email, phone_number,
employee_id emp_id
from hr.employees;
嘗試更新email和name
update hr.company_phone_book
 set name=’chen1, donny1’
where emp_id=100
create or replace trigger update_name_company_phone_book
instead of
update on hr.company_phone_book
begin
 update hr.employees
  set employee_id=:new.emp_id,
   first_name=substr(:new.name, instr(:new.name,’,’)+2),
   last_name= substr(:new.name,1,instr(:new.name,’,’)-1),
   phone_number=:new.phone_number,
   email=:new.email
 where employee_id=:old.emp_id;
end;
 
四、 系統事件觸發器
系統事件:數據庫啓動、關閉,服務器錯誤
create trigger ad_startup
 after startup
  on database
begin
 -- do some stuff
end;
/

五、 用戶事件觸發器
用戶事件:用戶登錄、註銷,create / alter / drop / analyze / audit / grant / revoke / rename / truncate / logoff
例子:記錄刪除對象
1. 日誌表
create table droped_objects(
 object_name varchar2(30),
 object_type varchar2(30),
 dropped_on date);
2.觸發器
create or replace trigger log_drop_trigger
 before drop on donny.schema
begin
 insert into droped_objects values(
  ora_dict_obj_name,  -- 與觸發器相關的函數
  ora_dict_obj_type,
  sysdate);
end;
/

3. 測試
create table drop_me(a number);
create view drop_me_view as select *from drop_me;
drop view drop_me_view;
drop table drop_me;
select *from droped_objects

禁用和啓用觸發器
alter trigger <trigger_name> disable;
alter trigger <trigger_name> enable;
事務處理:
在觸發器中,不能使用commit / rollback
由於ddl語句具備隱式的commit,因此也不容許使用
視圖:
dba_triggers
 
2007-5-7 22:30:08   
 發表評語»»»    
 2008-3-20 22:48:02    針對某個字段的觸發器
oracle沒有sqlserver那種機制,只能對字段進行比較判斷是否更新
create or replace trigger tri_t
 before update on t for each row
 when (new.bb<>old.bb)--比較BB字段是否更新
 begin
 :new.cc:=1;
 end;
 /

 2008-3-20 22:58:30    ORACLE觸發器語法
ORACLE觸發器語法
觸發器有兩種after和before,區別在於每次提交事務觸發一次和每一行數據的更新都觸發一次,使用時候要注意,以避免達不到目的
ORACLE產生數據庫觸發器的語法爲:
create [or replace] trigger 觸發器名 觸發時間 觸發事件
on 表名
[for each row]
pl/sql 語句
其中:
觸發器名:觸發器對象的名稱。因爲觸發器是數據庫自動執行
的,所以該名稱只是一個名稱,沒有實質的用途。
觸發時間:指明觸發器什麼時候執行,該值可取:
before---表示在數據庫動做以前觸發器執行;
after---表示在數據庫動做以後出發器執行。
觸發事件:指明哪些數據庫動做會觸發此觸發器:
insert:數據庫插入會觸發此觸發器;
update:數據庫修改會觸發此觸發器;
delete:數據庫刪除會觸發此觸發器。
表 名:數據庫觸發器所在的表。
for each row:對錶的每一行觸發器執行一次。若是沒有這一
選項,則只對整個表執行一次。
舉例:下面的觸發器在更新表auths以前觸發,目的是不容許在
週末修改表:
create trigger auth_secure
before insert or update or delete //對整表更新前觸發
on auths
begin
if(to_char(sysdate,'DY')='SUN'
RAISE_APPLICATION_ERROR(-20600,'不能在週末修改表auths');--觸發器中只能拋異常實現回滾
end if;
end   
相關文章
相關標籤/搜索