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会出现以下错误,如果我想达到我的要求应该怎么改这程序呢?

高手指点下!!!
展开
 我来答
wangzhiqing999
2010-09-30 · TA获得超过1.6万个赞
知道大有可为答主
回答量:7048
采纳率:100%
帮助的人:3261万
展开全部
关于 前面 执行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.其他字段);
陈学阳
2010-10-10 · TA获得超过2.8万个赞
知道大有可为答主
回答量:2.1万
采纳率:14%
帮助的人:5480万
展开全部
修改后如下:
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
本回答被网友采纳
已赞过 已踩过<
你对这个回答的评价是?
评论 收起
_李占涛_
2010-10-13
知道答主
回答量:13
采纳率:0%
帮助的人:6万
展开全部
楼上说的挺对的,你把before 换成after就可以:old.empno;
已赞过 已踩过<
你对这个回答的评价是?
评论 收起
收起 更多回答(1)
推荐律师服务: 若未解决您的问题,请您详细描述您的问题,通过百度律临进行免费专业咨询

为你推荐:

下载百度知道APP,抢鲜体验
使用百度知道APP,立即抢鲜体验。你的手机镜头里或许有别人想知道的答案。
扫描二维码下载
×

类别

我们会通过消息、邮箱等方式尽快将举报结果通知您。

说明

0/200

提交
取消

辅 助

模 式