Oracle PL/SQL之自定义函数的读一致性

本文通过一个具体的例子解释了在Oracle数据库中读一致性的问题。特别是在长时间运行的查询过程中,如果其他事务修改了数据,则可能导致自定义函数返回的结果与直接的SQL聚合函数结果不一致。文章还提供了使用SET TRANSACTION READ ONLY来避免此类问题的方法。

读一致性即确保查询的结果与发起查询的时刻的数据一致,不管在此查询期间其他事务有没有更改数据。

首先假设我们有如下函数用来根据部门id求该部门的总薪资:

CREATE OR REPLACE FUNCTION sum2(p_deptno IN NUMBER) RETURN NUMBER IS --PRAGMA AUTONOMOUS_TRANSACTION; l_ret NUMBER; BEGIN dbms_lock.sleep(5); --DBMS_BACKUP_RESTORE.SLEEP(5); dbms_output.put_line(systimestamp); SELECT SUM(sal) INTO l_ret FROM emp WHERE deptno = p_deptno; RETURN l_ret; END sum2;

然后我们开个session(s1),执行如下查询(q1):

SELECT deptno ,SUM(sal) ,sum2(deptno) ,systimestamp FROM emp GROUP BY deptno;

在q1执行的过程中,我们又开个新的session(s2),执行更新命令并提交:

SQL> update emp set sal=sal+1; 14 rows updated SQL> commit; Commit complete SQL>

接着回到s1,看q1的执行结果:

SQL> SELECT deptno 2 ,SUM(sal) 3 ,sum2(deptno) 4 ,systimestamp 5 FROM emp 6 GROUP BY deptno; DEPTNO SUM(SAL) SUM2(DEPTNO) SYSTIMESTAMP ------ ---------- ------------ -------------------------------------------------------------------------------- 30 9400 9406 24-JUN-11 08.31.19.722000 PM +08:00 20 10875 10880 24-JUN-11 08.31.19.722000 PM +08:00 10 8750 8753 24-JUN-11 08.31.19.722000 PM +08:00 SQL>

我们发现sum(sal)和sum2(deptno)的结果不一致。

当我们将自定义函数应用于sql语句中,而该sql又需要执行很长时间,并且在这段时间中恰好又有其他dml会更改该sql中的某些表时,这种不一致就会体现出来。

关于如何避免这种不一致,可以使用SET TRANSACTION READ ONLY,详细内容可参看我下面的这篇文章:

Oracle PL/SQL之SET TRANSACTION READ ONLY(事务隔离性)

http://blog.youkuaiyun.com/t0nsha/archive/2011/06/24/6566517.aspx

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值