`

oracle超出打开游标的最大数的原因和解决方案

 
阅读更多

处理超出打开游标的最大数异常(ORA-01000: maximum open cursors exceeded

ORA-01000_maximum_open_cursors_exceeded_

在执行如下代码时,经常会出现ora-01000: maximum open cursors exceeded异常
for(int i=0;i<balancelist.size();i++)
{
     prepstmt = conn.prepareStatement(sql[i]);
     prepstmt.setBigDecimal(1,nb.getRealCost());
     prepstmt.setString(2, adclient_id);
     prepstmt.setString(3, daystr);
     prepstmt.setInt(4, ComStatic.portalId);
     prepstmt.executeUpdate();
}

1. 检查数据库中的 OPEN_CURSORS 参数值。
Oracle 使用 init.ora 中的初始化参数 OPEN_CURSORS 指定一个会话一次最多可以拥有的游标数。缺省值为 50。要获得数据库中 OPEN_CURSORS 参数的值,可以使用以下查询:   
SQL> show parameter open_cursors;
NAME                                  TYPE         VALUE
------------------------------------ ----------- ---------------
open_cursors                          integer      300   


修改open_cursors

SQL> alter system set open_cursors=1000;

系统已更改。

SQL> commit;

提交完成。

SQL> show parameter open_cursors;

NAME                                 TYPE        VALUE
------------------------------------ ----------- ---------------------
open_cursors                         integer     1000


重要的是将 OPEN_CURSORS 的值设置得足够大,以避免应用程序用尽所有打开的游标。应用程序不同,该值也不同。即便会话打开的游标数未达OPEN_CURSORS 指定的数量(即设置的值高于实际需要的值),也不会增加系统开销。
2. 获取打开的游标数。
下面的查询按降序显示用户SCOTT为每个会话打开的游标数。  
SQL>

SELECT O.SID, s.OSUSER, s.MACHINe, count(1) num_curs
  FROM v$open_cursor o, v$session s
 WHERE o.USER_NAME = 'DB_YWBZ'
   AND O.SID = S.SID
 GROUP BY O.SID, s.OSUSER, s.MACHINE
 ORDER BY num_curs DESC;

 
    SID OSUSER        MACHINE            NUM_CURS
-----------------------------------------------------
        217            m1                 1000
         96            m2                 10
        411            m3                 10
         50           test                 9
请注意,v$open_cursor 可以跟踪会话中 PARSED  NOT CLOSED 的动态游标(使用 dbms_sql.open_cursor() 打开的游标)。它不会跟踪未经分析(但已打开)的动态游标。在应用程序中使用动态游标并不常见。本模式的前提是未使用动态游标。
3. 获取为游标执行的 SQL
使用在以上查询结果中找到的 SID 运行下面的查询:
SQL>

select q.sql_text
  from v$open_cursor o, v$sql q
  where q.hash_value=o.hash_value and o.sid = 1852;

 
SQL_TEXT
select * from empdemo where empid='212'
select * from empdemo where empid='321'
select * from empdemo where empid='947'
select * from empdemo where empid='527'
...
结果将显示正在连接上执行的查询。它提供了一个入手点,让您可以反向跟踪到打开游标的来源。


这样的错误很容易出现在Java代码中的主要原因是:Java代码在执行conn.createStatement() conn.prepareStatement()的时候,实际上都是相当与在数据库中打开了一个cursor。尤其是,如果你的 createStatementprepareStatement是在一个循环里面的话,就会非常容易出现这个问题。因为游标一直在不停的打开,而且没有关闭。
一般来说,我们在写Java代码的时候,createStatementprepareStatement都应该要放在循环外面,而且使用了这些 Statment后,及时关闭。最好是在执行了一次executeQueryexecuteUpdate等之后,如果不需要使用结果集(ResultSet)的数据,就马上将StatementPreparedStatement关闭。
对于出现ORA-01000错误这种情况,单纯的加大open_cursors并不是好办法,那只是治标不治本  实际上,代码中的隐患并没有解除。
而且,绝大部分情况下,open_cursors只需要设置一个比较小的值,就足够使用了,除非有非常特别的要求。
     如果你不使用连接池,那么就没有什么问题,一旦Connection关闭,数据库物理连接就被释放,所有相关Java资源也可以被GC回收了。
但是如果你使用连接池,那么请注意,Connection关闭并不是物理关闭,只是归还连接池,所以PreparedStatement ResultSet都被持有,并且实际占用相关的数据库的游标资源,在这种情况下,只要长期运行,往往就会报游标超出数据库允许的最大值的错误,导致程序无法正常访问数据库。
正确的代码,如下所示:
for(int i=0;i<balancelist.size();i++)
{
     prepstmt = conn.prepareStatement(sql[i]);
     prepstmt.setBigDecimal(1,nb.getRealCost());
     prepstmt.setString(2, adclient_id);
     prepstmt.setString(3, daystr);
     prepstmt.setInt(4, ComStatic.portalId);
     prepstmt.executeUpdate();
   prepstmt.close();
}
在执行了一次executeQueryexecuteUpdate等之后,如果不需要使用结果集(ResultSet)的数据,就马上将StatementPreparedStatement关闭。

分享到:
评论

相关推荐

    超出打开游标的最大数的原因和解决方案

    超出打开游标的最大数的原因和解决方案 在 Oracle 数据库中,游标(cursor)是一种临时工作区,用于存储查询结果。每个会话可以打开多个游标,但存在一个限制,即 OPEN_CURSORS 参数指定的最大游标数。如果应用程序...

    Oracle数据库游标连接超出解决方案

    3.获取打开的游标数 select o.sid, osuser, machine, count(*) num_curs from v$open_cursor o, v$session s where user_name = 'AF651_U8' and o.sid = s.sid group by o.sid, osuser, machine order b

    oracle异常(最全异常收集)

    ### Oracle异常详解 #### ORA-0001:DUP_VAL_ON_INDEX ...了解这些异常的原因和解决方案有助于提高数据库应用的稳定性和效率。对于每一种异常,都应该采取相应的预防措施和处理策略,确保应用程序能够平稳运行。

    Oracle错误码大全

    错误码通常会附带一个简短的错误描述,提供关于问题原因和可能解决方案的线索。 在Oracle 8i中,常见的错误码包括但不限于: 1. ORA-01000: 达到最大打开游标数 - 表示应用程序打开了过多的游标,超过了系统允许的...

    类型长度大于最大值(Java调用存储过程)

    标题与描述均指出的问题是“类型长度大于最大值(Java调用存储过程)”,这通常意味着在使用Java程序调用Oracle数据库的存储过程时,某个字段或参数的长度超过了数据库定义的最大长度。这种情况可能由多种因素引起,...

    oracle常见Exception

    为了更好地理解和处理这些异常,本篇文章将详细介绍Oracle中常见的几种异常及其解决方案。 #### CURSOR_ALREADY_OPEN (ORA-06511) **异常描述:** 当尝试打开一个已经处于打开状态的游标时,系统会抛出`CURSOR_...

    如何避免JDBC引起的内存溢出情况

    #### 二、针对不同数据库的解决方案 **1. MySQL** 对于MySQL,可以通过修改JDBC连接字符串中的参数来控制结果集的读取方式。具体来说,可以在连接字符串中添加`useCursorFetch=true`和`defaultFetchSize=100`两个...

Global site tag (gtag.js) - Google Analytics