Oracle 触发器执行问题
我创建了表audit_alter有字段auseractionadate来存储deleteupdateinsert操作情况还创建了表standby_empcreatesta...
我创建了表audit_alter 有字段auser action adate 来存储delete update insert操作情况
还创建了表standby_emp create standby_emp as select * from emp where 1=0;用来存放意外删除的数据。 触发器代码如下:
create or replace trigger unexpect_recoord
before delete or update or insert on emp
for each row
declare
a audit_alter.action%type;
begin
if deleting
then a:='delete';
insert into standby_emp (select * from emp where emp.empno=:old.empno);
end if;
if inserting
then a:='insert';
end if;
if updating
then a:='update';
end if;
insert into audit_alter values(user,a,sysdate);
end;
但是我insert 一条记录后表audit_alter 显示有2条 insert 操作;
update 6条数据后,表audit_alter显示有12条 update 操作
然后执行delete 操作 显示 以下错误:
delete from emp where ename='d'
ORA-04091: 表 ZHOU.EMP 发生了变化, 触发器/函数不能读它
ORA-06512: 在 "ZHOU.UNEXPECT_RECOORD", line 6
ORA-04088: 触发器 'ZHOU.UNEXPECT_RECOORD' 执行过程中出错
谁告诉我为什么表audit_alter里显示的insert update 操作是执行的2倍啊 ,
还有是为什么delete会出现以下错误,如果我想达到我的要求应该怎么改这程序呢?
高手指点下!!! 展开
还创建了表standby_emp create standby_emp as select * from emp where 1=0;用来存放意外删除的数据。 触发器代码如下:
create or replace trigger unexpect_recoord
before delete or update or insert on emp
for each row
declare
a audit_alter.action%type;
begin
if deleting
then a:='delete';
insert into standby_emp (select * from emp where emp.empno=:old.empno);
end if;
if inserting
then a:='insert';
end if;
if updating
then a:='update';
end if;
insert into audit_alter values(user,a,sysdate);
end;
但是我insert 一条记录后表audit_alter 显示有2条 insert 操作;
update 6条数据后,表audit_alter显示有12条 update 操作
然后执行delete 操作 显示 以下错误:
delete from emp where ename='d'
ORA-04091: 表 ZHOU.EMP 发生了变化, 触发器/函数不能读它
ORA-06512: 在 "ZHOU.UNEXPECT_RECOORD", line 6
ORA-04088: 触发器 'ZHOU.UNEXPECT_RECOORD' 执行过程中出错
谁告诉我为什么表audit_alter里显示的insert update 操作是执行的2倍啊 ,
还有是为什么delete会出现以下错误,如果我想达到我的要求应该怎么改这程序呢?
高手指点下!!! 展开
展开全部
关于 前面 执行2次的,我还不确定是什么原因
对于 DELETE 出错的情况, 是取值的方式不正确
请看下面这个 DELETE 的触发器的例子:
SQL> CREATE OR REPLACE TRIGGER BeforeDeleteTest
2 BEFORE DELETE ON test_trigger_table
3 FOR EACH ROW
4 BEGIN
5 dbms_output.put_line('BEFORE DELETE');
6 dbms_output.put_line('Old Name = ' || :old.name);
7 dbms_output.put_line('New Name = ' || :new.name);
8 END;
9 /
Trigger created.
SQL>
SQL> DELETE FROM test_trigger_table WHERE id = 1;
BEFORE DELETE
Old Name = XYZ
New Name =
1 row deleted.
你要直接把所有的字段 都写成 :old.......的方式。
也就是
insert into standby_emp
VALUES(:old.empno, :old.其他字段);
对于 DELETE 出错的情况, 是取值的方式不正确
请看下面这个 DELETE 的触发器的例子:
SQL> CREATE OR REPLACE TRIGGER BeforeDeleteTest
2 BEFORE DELETE ON test_trigger_table
3 FOR EACH ROW
4 BEGIN
5 dbms_output.put_line('BEFORE DELETE');
6 dbms_output.put_line('Old Name = ' || :old.name);
7 dbms_output.put_line('New Name = ' || :new.name);
8 END;
9 /
Trigger created.
SQL>
SQL> DELETE FROM test_trigger_table WHERE id = 1;
BEFORE DELETE
Old Name = XYZ
New Name =
1 row deleted.
你要直接把所有的字段 都写成 :old.......的方式。
也就是
insert into standby_emp
VALUES(:old.empno, :old.其他字段);
展开全部
修改后如下:
create or replace trigger tri_borrow
after insert on borrow
for each row
declare
vn_count int;
begin
select count(1) into vn_count from books
where :new.bno = books.bno and books.bname = 'database';
if vn_count>0 then
insert into borrow_save(cno,bno,rdate) values (:new.cno,:new.bno,:new.rdate);
end if;
end;
原因一:不能直接写select from 要定义变量 通过select into 变量 from
原因二:在加了触发器的表在触发过程中不能对该表进行操作包括查询。所以需要将两表关联中的borrow去掉,改为用:new.bno来做约束条件。
如果对您有帮助,请记得采纳为满意答案,谢谢!祝您生活愉快!
vaela
create or replace trigger tri_borrow
after insert on borrow
for each row
declare
vn_count int;
begin
select count(1) into vn_count from books
where :new.bno = books.bno and books.bname = 'database';
if vn_count>0 then
insert into borrow_save(cno,bno,rdate) values (:new.cno,:new.bno,:new.rdate);
end if;
end;
原因一:不能直接写select from 要定义变量 通过select into 变量 from
原因二:在加了触发器的表在触发过程中不能对该表进行操作包括查询。所以需要将两表关联中的borrow去掉,改为用:new.bno来做约束条件。
如果对您有帮助,请记得采纳为满意答案,谢谢!祝您生活愉快!
vaela
本回答被网友采纳
已赞过
已踩过<
评论
收起
你对这个回答的评价是?
展开全部
楼上说的挺对的,你把before 换成after就可以:old.empno;
已赞过
已踩过<
评论
收起
你对这个回答的评价是?
推荐律师服务:
若未解决您的问题,请您详细描述您的问题,通过百度律临进行免费专业咨询