ora-01403 no data found error,ora-04088 执行触发器时出错

Posted

技术标签:

【中文标题】ora-01403 no data found error,ora-04088 执行触发器时出错【英文标题】:ora-01403 no data found error,ora-04088 error during execution of trigger 【发布时间】:2014-04-30 14:21:51 【问题描述】:

嗨,我有这个触发器

create or replace TRIGGER trg_cust
after insert or update on nbty_customer
referencing old as old new as new
for each row
declare
pragma autonomous_transaction;
V_REQ VARCHAR2(10);
begin 

 IF(:NEW.cust_sal<1000)
        THEN
        select :NEW.cust_sal+:OLD.cust_sal INTO V_REQ FROM nbty_customer
        where cust_id=:old.cust_id;

    ELSE
    SELECT :NEW.cust_sal-:OLD.cust_sal INTO V_REQ FROM nbty_customer where cust_id=:old.cust_id;
    end if;

merge into nbty_cache_cust 
    using
       (select distinct :new.cust_id cust_id, :new.cust_name cust_name,V_REQ V_REQ
       from nbty_customer) b
       on (cust_nbty_id = b.cust_id)
        when matched then
       update set cust_nbty_name = b.cust_name, cust_nbty_sal = b.V_REQ 
       when not matched then
      insert (cust_nbty_id, cust_nbty_name, cust_nbty_sal)
      values (b.cust_id, b.cust_name, b.V_REQ);
      commit;
      end;

哪个编译正确,但是当在表 nbty_customer 上完成插入时,例如

insert into nbty_customer values('2','abc','200')

在执行触发器时抛出 ora-01403 no data found 错误,ora-04088 错误 请帮忙,我无法弄清楚问题是什么?

【问题讨论】:

没有数据我们无法判断。事实是,有一个 select 没有产生任何结果,你没有抓住。 【参考方案1】:

问题是由于 select into 子句返回“未找到数据”,因此我删除了 select into 并直接使用绑定变量:

IF(:new.cust_sal<1000)
then
  v_req:=nvl(:new.cust_sal,0)+nvl(:old.cust_sal,0);
else
  v_req:=nvl(:new.cust_sal,0)-nvl(:old.cust_sal,0);
end if;

这解决了问题。

【讨论】:

【参考方案2】:

我刚刚评论了,然后我注意到了:

select :NEW.cust_sal+:OLD.cust_sal INTO V_REQ FROM nbty_customer
    where cust_id=:old.cust_id;

insert 上,这个select 将失败,因为没有:old.cust_id。请改用:new.cust_id。 (另外,:old.cust_sal 也将为空)

【讨论】:

是的,我用过 :new.cust_id 仍然抛出同样的错误,抱歉这么晚才回复 我尝试使用 select nvl(:NEW.cust_sal,0)+nvl(:OLD.cust_sal,0) INTO V_REQ FROM nbty_customer where cust_id=:new.cust_id;但无济于事,它仍然抛出同样的错误,请帮助 能否在问题中添加完整的插入语句,包括列名? 插入 nbty_customer(cust_id,cust_name,cust_sal) 值 ('111','abc,'1000'); 这将进入无效的 else 分支。那时还没有客户。【参考方案3】:

在您的触发器正文中,您正在尝试选择新行的数据:

 SELECT :NEW.cust_sal-:OLD.cust_sal INTO V_REQ FROM nbty_customer where cust_id=:old.cust_id;

但是您将触发器主体声明为PRAGMA AUTONOMOUS_TRANSACTION。因此,您的触发器首先看不到触发触发器执行的更改。

在触发器中使用AUTONOMOUS_TRANSACTION从不是个好主意;请参阅Tom Kyte on Triggers,详细了解为什么要避免这种情况。

如果您出于性能原因需要缓存数据,我建议您改用物化视图。如果需要,您可以将其设置为REFRESH ON COMMIT

create materialized view mv_cache
refresh on commit 
as 
select distinct cust_id cust_id, cust_name cust_name, V_REQ 
from nbty_customer

【讨论】:

以上是关于ora-01403 no data found error,ora-04088 执行触发器时出错的主要内容,如果未能解决你的问题,请参考以下文章

为啥 no_data_found ORA-01403 在 Oracle 中是一个异常?

NO_DATA_FOUND ORACL NVL函数,当第一个为空时显示第二个参数值

OGG replicat错误 ORA-01403及解决方案

ORA-01403:未找到数据 ORA-06512:在第 8 行 01403。00000 -“未找到数据”

ORA-01403 找不到数据错误

PLSQL 触发器 ORA 01403 未找到数据