Oracle中触发器(2)
admin
2023-05-08 15:22:03
0

   上一篇对触发器的一些基本知识有了了解,在这一篇操作进行验证

before触发器

[oracle@test ~]$ sqlplus / as sysdba

SQL> create user trigger_test identified by 123456 ;
SQL> grant create session to trigger_test ;
SQL> grant create table to trigger_test ;
SQL> grant dba to trigger_test ;
SQL> grant resource to trigger_test ;
SQL> grant create sequence to trigger_test ;

[oracle@test ~]$ sqlplus trigger_test/123456

创建测试表student

SQL> create table student(STUDENT_ID NUMBER(8),name varchar2(15),CREATED_BY VARCHAR2(30) not null,CREATED_DATE DATE not null,MODIFIED_BY VARCHAR2(30) not null,MODIFIED_DATE DATE not null)

创建序列student_id_seq

create sequence student_id_seq
minvalue 1
maxvalue 9999999999999999999999999999
start with 100
increment by 1;
SQL> select * from student ;

no rows selected


SQL> select * from cat ;

TABLE_NAME		       TABLE_TYPE
------------------------------ -----------
STUDENT 		       TABLE
STUDENT_ID_SEQ		       SEQUENCE
create or replace trigger student_before_insert
before insert on student
for each row 
declare
  v_student_id student.student_id%type ;
begin
  select student_id_seq.nextval
  into v_student_id
  from dual ;
  
  :new.student_id := v_student_id ;
  :new.created_by := user ;
  :new.created_date := sysdate ;
  :new.modified_by := user ;
  :new.modified_date := sysdate ;
  
end;
SQL> insert into student (name) values ('mjt');

SQL> commit ;

SQL> select * from student ;

STUDENT_ID NAME 	   CREATED_BY			  CREATED_DATE
---------- --------------- ------------------------------ -------------------
MODIFIED_BY		       MODIFIED_DATE
------------------------------ -------------------
       100 mjt		   TRIGGER_TEST 		  2015-07-29 20:51:19
TRIGGER_TEST		       2015-07-29 20:51:19

SQL> insert into student (name) values ('cxq') ;

1 row created.

SQL> select * from student ;

STUDENT_ID NAME 	   CREATED_BY			  CREATED_DATE
---------- --------------- ------------------------------ -------------------
MODIFIED_BY		       MODIFIED_DATE
------------------------------ -------------------
       100 mjt		   TRIGGER_TEST 		  2015-07-29 20:51:19
TRIGGER_TEST		       2015-07-29 20:51:19

       101 cxq		   TRIGGER_TEST 		  2015-07-29 21:00:54
TRIGGER_TEST		       2015-07-29 21:00:54

after触发器

SQL> create table record(table_name varchar2(30),transaction_name varchar2(10),transaction_user varchar2(30),transaction_date date);

这个表被用来记录数据库中不同表的信息,如,可以记录谁从student表中删除或者更新数据,以及记录时间。

下面的触发器针对对student表的更新或者删除操作,在此之后进行触发

create or replace trigger student_aud
  after update or delete on student
declare
  v_type varchar2(10);
begin
  if updating then
    v_type := 'UPDATE';
  
  elsif deleting then
    v_type := 'DELETE';
  end if;

  update trigger_test.record
     set transaction_user = user, 
     transaction_date = sysdate
   where table_name = 'student'
     and transaction_name = v_type;

  if sql%notfound then
    insert into trigger_test.record values ('student', v_type, user, sysdate);
  end if;

end;
SQL> select * from record ;

no rows selected

SQL> select * from student ;

STUDENT_ID NAME 	   CREATED_BY			  CREATED_DATE
---------- --------------- ------------------------------ -------------------
MODIFIED_BY		       MODIFIED_DATE
------------------------------ -------------------
       100 mjt		   TRIGGER_TEST 		  2015-07-29 20:51:19
TRIGGER_TEST		       2015-07-29 20:51:19

       101 cxq		   TRIGGER_TEST 		  2015-07-29 21:00:54
TRIGGER_TEST		       2015-07-29 21:00:54


SQL> update student set name = 'somebody' where name = 'mjt' ;

1 row updated.

SQL> commit ;

Commit complete.

SQL> select * from student ;

STUDENT_ID NAME 	   CREATED_BY			  CREATED_DATE
---------- --------------- ------------------------------ -------------------
MODIFIED_BY		       MODIFIED_DATE
------------------------------ -------------------
       100 somebody	   TRIGGER_TEST 		  2015-07-29 20:51:19
TRIGGER_TEST		       2015-07-29 20:51:19

       101 cxq		   TRIGGER_TEST 		  2015-07-29 21:00:54
TRIGGER_TEST		       2015-07-29 21:00:54

SQL> select * from record ;

TABLE_NAME		       TRANSACTIO TRANSACTION_USER
------------------------------ ---------- ------------------------------
TRANSACTION_DATE
-------------------
student 		       UPDATE	  TRIGGER_TEST
2015-07-29 21:50:46


SQL> delete student where name = 'somebody' ;

1 row deleted.

SQL> commit ;

Commit complete.

SQL> select * from student ;

STUDENT_ID NAME 	   CREATED_BY			  CREATED_DATE
---------- --------------- ------------------------------ -------------------
MODIFIED_BY		       MODIFIED_DATE
------------------------------ -------------------
       101 cxq		   TRIGGER_TEST 		  2015-07-29 21:00:54
TRIGGER_TEST		       2015-07-29 21:00:54

SQL> select * from record ;

TABLE_NAME		       TRANSACTIO TRANSACTION_USER
------------------------------ ---------- ------------------------------
TRANSACTION_DATE
-------------------
student 		       UPDATE	  TRIGGER_TEST
2015-07-29 21:50:46

student 		       DELETE	  TRIGGER_TEST
2015-07-29 21:56:08


SQL> alter trigger student_aud disable ;

Trigger altered.

SQL> update student set name = 'mjt' where name = 'cxq';

1 row updated.

SQL> commit ;

Commit complete.

SQL> select * from student ;

STUDENT_ID NAME 	   CREATED_BY			  CREATED_DATE
---------- --------------- ------------------------------ -------------------
MODIFIED_BY		       MODIFIED_DATE
------------------------------ -------------------
       101 mjt		   TRIGGER_TEST 		  2015-07-29 21:00:54
TRIGGER_TEST		       2015-07-29 21:00:54

SQL> select * from record ;

TABLE_NAME		       TRANSACTIO TRANSACTION_USER
------------------------------ ---------- ------------------------------
TRANSACTION_DATE
-------------------
student 		       UPDATE	  TRIGGER_TEST
2015-07-29 21:50:46

student 		       DELETE	  TRIGGER_TEST
2015-07-29 21:56:08

禁用触发器之后,在student表上的update操作不再触发产生记录到record表

对应启用

SQL> alter trigger student_aud enable ;

Trigger altered.

3.自治事务


自治事务是由其他事务(通常被称为主事务)发起的独立事务,自治事务也许会执行多个dml语句,并且提交或者回滚,而不会提交或者回滚主事务执行的dml语句。


假如希望即使主事务失败,仍旧能够记录审计数据,这种情况下,主事务是面向

表的update或者delete,需要定义可以独立于主事务进行提交的自治事务。


定义主事务,需要使用autonomous_transaction编译指令在语句块的声明部分

declare 
pragma autonomous_transaction

commit ;

create or replace trigger student_aud
  after update or delete on student  
declare
  v_type varchar2(10);
  pragma autonomous_transaction ;
begin
  if updating then
    v_type := 'UPDATE';
  
  elsif deleting then
    v_type := 'DELETE';
  end if;

  update trigger_test.record
     set transaction_user = user, 
     transaction_date = sysdate
   where table_name = 'student'
     and transaction_name = v_type;

  if sql%notfound then
    insert into trigger_test.record values ('student', v_type, user, sysdate);
  end if;
  commit ;
end;

如果当前更新或者删除表student中的内容,无论成功或者失败,都会在record表中记录当前的操作。不足的是,record表中最多只能记录两条数据,只是当前最新操作的时间以及操作者。

上一篇:oracle基本操作

下一篇:Rman--备份命令

相关内容

热门资讯

特稿|拉长合作清单 贡献建设性... 新华社北京5月13日电 题:拉长合作清单 贡献建设性力量——美国商学界人士瞩望美中经贸关系互利共赢新...
中国量子计算新突破!“九章四号... 记者从中国科学技术大学获悉,由该校潘建伟院士领衔的科研团队联合国内多家科研机构、大学,近期成功研制出...
跳河救人的外卖小哥找到了! 外... 5月12日下午5时许,漯河市郾城区孟庙镇幸福渠河堤旁,57岁的甘女士蹲在河边打水,准备回家给鱼换水,...
今年以来,越来越多美国交流团来... 4月,数十名美国犹他州青少年来豫参加2026年YES项目交流活动。图为美国青少年在郑州体验书法项目。...
“打工机器人”亮相郑州街头 机器人服务员“小盖”在郑州街边的一零售店工作。 王磊 摄机器人当服务员,在街头卖咖啡——这不是科幻电...
打响“河南服务”品牌丨盾构机有... 【开栏的话】为深入贯彻落实全省服务业大会精神,本报即日起开设“打响‘河南服务&rsquo...
一季度我国数字产业收入9.5万... 【大河财立方消息】5月14日,工信部发布的数据显示,一季度,我国数字产业实现良好开局,行业利润大幅改...
一体推进整治形式主义为基层减负... 形式主义实质是主观主义、功利主义,根源是政绩观错位、责任心缺失。当前,各地以深化“六个纠治”为抓手,...
5月上旬汽油柴油价格环比继续下... 【大河财立方消息】 5月14日,国家统计局发布2026年5月上旬流通领域重要生产资料市场价格变动情况...
河南信阳凌晨通报:常某朋(男,... 2026年5月13日21时43分许,我市浉河区发生一起道路交通事故。经查,常某朋(男,40岁)驾驶私...