经过检查,原来这个对象是一个临时表。我们知道,临时表对象平时是不存在数据的。只有当一个会话使用临时表,并向表中插入数据后,oracle才会在临时表空间上创建它的数据对象。临时表数据之所以只被所调用会话看到,是实际上是在每个会话中创建了一个单独的数据对象,有各自的数据对象标号。因此尽管是同一个临时表,每个会话只是copy一个表结构,而创建了不同的数据对象,这样,会话之间就不会有数据干扰。而在一个会话中,对临时表数据对象的处理跟普通数据对象处理基本相同,其中就包括临时表对象在事务中的数据改动也会有回滚信息的产生。
回到我们的问题中。通过V$SESSION和V$SQL_AREA查到,这些会话都是调用了一个PLSQL函数,而且都是通过java调用的。
Review代码,终于发现潜在问题了:这个函数的结果是返回一个游标,而游标恰恰关联了这张临时表。
INSERT INTO TMP_CNT_GRP …
SELECT…
… …
open v_cursor FOR
select TMP.CDE,
CAR.ID,
CAR.NME,
COUNT(DISTINCT TMP.NUM) TOTAL_CNT
from TMP_CNT_GRP TMP,
CSS_CAR CAR
WHERE TMP.ID = CAR.ID
GROUP BY TMP.CDE, CAR.ID, CAR.NME;
RETURN v_cursor;
DELETE TMP_CNT_GRP;
(这段代码其实还存在一个问题,也就是最后的DELETE语句根本不会被调用)
从这段代码中可以看到,实际上在整个函数当中,临时表的数据根本不会被释放;而且也没有提交和回滚事务(尽管这是一个会话级的临时表)。占用的回滚段也不会被释放。这就存在这样的潜在问题,如果调用者不关闭会话或提交/回滚事务的话,它所占用的回滚段就不会被释放。事实上,经过让java开发人员检查代码,果然发现客户端在打开会话后,就没有关闭,知道客户端本身结束。
最后的解决办法:
1、因为这是一个会话级的临时表,数据在事务提交后继续保留,因此在PLSQL函数中的insert语句后加上commit;
2、Java代码在使用完游标后关闭会话。
我们一直都在努力坚持原创.......请不要一声不吭,就悄悄拿走。
我原创,你原创,我们的内容世界才会更加精彩!
【所有原创内容版权均属TechTarget,欢迎大家转发分享。但未经授权,严禁任何媒体(平面媒体、网络媒体、自媒体等)以及微信公众号复制、转载、摘编或以其他方式进行使用。】
微信公众号

TechTarget
官方微博

TechTarget中国
作者
相关推荐
-
Notre Dame对云端SQL Server性能基准的探索实践
确立SQL Server的性能基准,对于云端迁移来说是至关重要的第一步,一位来自于University of Notre Dame 的DBA表示,他正在试图通过数据库监控软件,找出SQL server的性能基准。
-
DBA必须掌握的数据库恢复管理技术
如果没有备份副本,数据库管理员就无法还原数据库,所以DBA在恢复之前倾向于考虑备份是合乎逻辑的。 但是,对我来说,这种逻辑一直是错误的。
-
控制合约 不再畏惧Oracle
许多公司都与Oracle有无限制授权协议,他们害怕离开这个协议,所以就证明他们在使用Oracle的软件,即使因为需求单独购买部分授权许可也可能总体是省钱的。
-
如何应对Oracle EBS实施中的六个挑战?
在18个月的时间里,Vitamix启动运行了Oracle电子商务套件(E-Business Suite,EBS),而且Vitamix还对诸如Oracle ATG Web Commerce等所有页面属性进行了重新整理,并实现了全球数据中心。