`
onlyor
  • 浏览: 381031 次
  • 性别: Icon_minigender_1
  • 来自: 南京
社区版块
存档分类
最新评论

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

    博客分类:
  • jdbc
阅读更多

在执行如下代码时,经常会出现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, osuser, machine, count(*) num_curs
   2   from v$open_cursor o, v$session s
   3   where user_name = 'SCOTT' and o.sid=s.sid
   4   group by o.sid, osuser, machine
   5 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
   2   from v$open_cursor o, v$sql q
   3   where q.hashvalue=o.hash_value and o.sid = 217;
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关闭。

分享到:
评论

相关推荐

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

    如果应用程序打开的游标数超过这个限制,会引发 ORA-01000: maximum open cursors exceeded 异常。下面将探讨超出打开游标的最大数的原因和解决方案。 原因 应用程序打开的游标数超过 OPEN_CURSORS 参数指定的最大...

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

    解决疑难问题,针对偏僻的问题:oracle超出打开游标的最大数的原因和解决方案,希望能对大家操作数据库有用处

    oracle+ora-各种常见java.sq

    - `java.sql.SQLException: ORA-01000: maximum open cursors exceeded`: 当打开的游标超过数据库允许的最大值时会出现此错误。检查代码中游标的关闭情况,避免资源泄漏。 8. **锁冲突** - `java.sql....

    oracle常见错误号

    1. ORA-01002:fetch out of sequence - 当游标没有正确地按照执行顺序进行操作时,比如在未调用`OPEN`、`FETCH`或`CLOSE`之前尝试`FETCH`,会出现这个错误。确保在执行`FETCH`前先调用`OPEN`。 2. ORA-00957:...

    Oracle出现超出打开游标最大数的解决方法

    当程序在处理大量数据或者进行复杂操作时,可能会遇到“超出打开游标最大数”的错误,即ORA-01000异常。这个错误通常表明数据库当前打开的游标数量超过了`open_cursors`参数所设定的最大值。 首先,我们需要理解为...

    Oracle常用异常代码

    当尝试打开一个已经打开的游标时触发此异常。 **解决方法:** - 在打开新的游标之前关闭已有的游标。 - 确保游标的状态管理正确,避免重复打开同一游标。 #### 5. DUP_VAL_ON_INDEX (ORA-00001) **异常描述:** 当...

    ORACLE技术文档\oracle cursor 游标.doc

    当一个应用程序尝试打开过多的游标时,可能会遇到`ORA-01000: maximum open cursors exceeded`错误,这表明单个用户尝试打开的游标数量超过了系统允许的最大值。 `ORA-01000`错误的主要原因通常是程序设计不当,...

    Oracle错误代码集合[归类].pdf

    20. ORA-17021: 定义丢失,通常意味着在处理结果集或游标时丢失了必要的信息。 21. ORA-17022: 在索引处定义丢失,可能是在尝试访问未定义的索引位置。 22. ORA-17023: 不支持的特性,表示尝试使用Oracle数据库不...

    Oracle错误代码集合[参考].pdf

    17. ORA-17070 至 ORA-17094: 包括数据大小超出限制、VARRAY溢出、操作不支持、值冲突以及并发级别和结果类型的问题。 18. ORA-17095 至 ORA-17099: 这些错误涵盖了各种内部错误、不允许的操作、创建结果集失败以及...

    预定义异常

    3. **异常传播**:如果某个过程无法处理异常,则可以通过`RAISE`语句将其传递给调用方。 #### 四、小结 通过了解和掌握这些预定义异常,开发者可以在PL/SQL程序中实现更健壮的异常处理机制,从而提高应用程序的稳定...

    oracle异常(最全异常收集)

    - **异常说明**:游标已打开。 - **常见原因**: - 尝试在同一作用域内多次打开同一个游标。 - 未正确关闭游标。 - **解决方案**: - 确保每个游标仅在一个作用域内被打开一次。 - 在使用完游标后立即关闭它。 ...

    Oracle错误码大全

    1. ORA-01000: 达到最大打开游标数 - 表示应用程序打开了过多的游标,超过了系统允许的最大值。解决方法是优化代码,关闭不再使用的游标,或者增加`OPEN_CURSORS`初始化参数的值。 2. ORA-01422: 指定的行数超出...

    Oracle 游标使用大全

    9. **游标处理异常** - 应适当地处理游标相关的异常,例如NO_DATA_FOUND和TOO_MANY_ROWS。 10. **游标最佳实践** - 及时关闭游标以释放系统资源。 - 避免在循环中打开和关闭游标,尽可能减少游标操作次数。 - ...

    Oracle数据库常见问题诊断-常见错误篇

    - **ORA-01000:** 当达到 `INIT.ORA` 文件中定义的最大打开游标数 (`OPEN_CURSORS`) 时,会出现此错误。 **解决办法:** - 增加 `INIT.ORA` 文件中的 `OPEN_CURSORS` 值,例如将其设为 200 或更高。 - 如果应用...

    oracle错误报告.docx

    在处理Oracle数据库时,可能会遇到各种错误,这些错误通常以ORA-错误代码的形式出现。以下是一些常见的Oracle错误及其详细解释: 1. ORA-00910: 指定的长度太长。这个错误通常发生在尝试创建或更新具有超过最大长度...

    SQLException

    `SQLException`是Java中处理所有数据库错误的标准异常类,它继承自`java.lang.Exception`。本文将围绕`SQLException`这一主题,对数据库操作过程中可能遇到的一些常见异常情况进行详细解析。 #### 二、SQLException...

    ora-exploits-evilcursor:Oracle Evil 游标注入漏洞利用库 - Perl 和 SQL 版本

    ora-exploits-evilcursor Oracle Evil 游标注入漏洞利用存储库 - Perl 和 SQL 版本SQL  bunkerview.sql  ctxsys-drvxtabc-create_tablesV2.sql  dbms_cdc_subscribeV2.sql  dbms_meta_get_ddlV2.sql  kupm...

    Oracle数据泵(Data Dump)使用过程当中经常会遇到一些奇奇怪怪的错误案例

    ORA-39065 表示在DISPATCH过程中遇到了未预期的主进程异常,而ORA-44002则意味着对象名无效。这个错误可能是由于内部错误或配置问题导致的。解决这个问题的一种方法是运行 `$ORACLE_HOME/rdbms/admin` 目录下的 `...

    Oracle错误一览表

    当一个会话达到其资源限制时,如最大打开游标数、最大连接时间等,会抛出此错误。这通常意味着会话配置需要调整,或者系统资源不足。 #### ORA-00021: Instance recovery in progress 在实例恢复过程中执行操作时...

Global site tag (gtag.js) - Google Analytics