trigger中的自治事务

 

这是来自CSDN里,网友问过的一个帖子,

 

网友也在我的blog里提过这样的问题

 

问题如下

 

表 KSOURCE.USERINFO 发生了变化, 触发器/函数不能读它
UserInfo(UserID,UserName,DepartID,SerialNum,…)
对于一个部门(DepartID相同)下的用户,我要让其序号SerialNum保持连续,就建立了一个触发器:

 

引用:
CREATE OR REPLACE TRIGGER “KSOURCE”.TRUSERINFO_INSERT
AFTER
INSERT ON “KSOURCE”.”USERINFO”
REFERENCING OLD AS OLD NEW AS NEW
FOR EACH ROW
Declare
v_SerialNum Number;
Begin
Select Count(*) Into v_SerialNum From UserInfo Where DepartID=:New.DepartID;
if (:New.SerialNum v_SerialNum) Then
Update UserInfo Set SerialNum=v_SerialNum Where UserID=:New.UserID;
end if;
End;


这样,如果用户输入的SerialNum不连续了,自动改成连续的,但Insert时却出现如下提示:
表 KSOURCE.USERINFO 发生了变化, 触发器/函数不能读它

这里为什么会出错了,这里的sql代码一点问题都没有,问题就出在是写在trigger里了。而且是for each row的trigger
行级的触发器代码中不能再操作该表,包括select,默认的在写TRIGGER的时候把本表锁死,不允许对其进行操作,

 

解决的方式

就是这里可以通过一些AUTONOMOUS_TRANSACTION(自治事务)来解决

 

或者是用表级别的trigger和行级trigger搭配,行级trigger做标记,表级别的trigger来做操作。配合着package里的临时变量存储标记。

 

一下是简单的处理方式,自治事务

引用:
CREATE OR REPLACE TRIGGER “KSOURCE”.TRUSERINFO_INSERT
AFTER
INSERT ON “KSOURCE”.”USERINFO”
REFERENCING OLD AS OLD NEW AS NEW
FOR EACH ROW
Declare
v_SerialNum Number;
PRAGMA AUTONOMOUS_TRANSACTION;
Begin
Select Count(*) Into v_SerialNum From UserInfo Where DepartID=:New.DepartID;
if (:New.SerialNum v_SerialNum) Then
Update UserInfo Set SerialNum=v_SerialNum Where UserID=:New.UserID;
end if;
commit;
End;


你试试这个解决方式叻,不过对于你这case,确实比较复杂,在你delete一条中间的数据时,确实还要控制其他的serialnum也随之发生变化。这样才能保持连续
由于是自治形式的,所以在你的程序块里,最后需要加上commit;

 




 

 

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

inthirties

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值