JAVA报游标超出最大连接数

来源:互联网 发布:java代码动态分析工具 编辑:程序博客网 时间:2024/05/16 23:56
java.sql.SQLException: ORA-01000: maximum open cursorsexceeded
//游标超出最大连接数

       atoracle.jdbc.dbaccess.DBError.throwSqlException(DBError.java:180)
       at oracle.jdbc.ttc7.TTIoer.processError(TTIoer.java:208)
       at oracle.jdbc.ttc7.Oopen.receive(Oopen.java:118)
       at oracle.jdbc.ttc7.TTC7Protocol.open(TTC7Protocol.java:472)
       atoracle.jdbc.driver.OracleStatement.<init>(OracleStatement.java:492)
       atoracle.jdbc.driver.OracleStatement.<init>(OracleStatement.java:511)
       atoracle.jdbc.driver.OraclePreparedStatement.<init>(OraclePreparedStatement.java:209)
       atoracle.jdbc.driver.OraclePreparedStatement.<init>(OraclePreparedStatement.java:192)
       atoracle.jdbc.driver.OracleConnection.privatePrepareStatement(OracleConnection.java:778)
       atoracle.jdbc.driver.OracleConnection.prepareStatement(OracleConnection.java:634)
       at portal.base.Sqlca.setSql(Sqlca.java:289)
       at portal.busi.toMusicBox.process(toMusicBox.java:271)
       at portal.busi.toMusicBox.main(toMusicBox.java:984)

 

解决:
报的异常是游标超出最大连接数;

经过查看代码发现在执行SQL的时候用的是prepareStatement。
实际上都是相当与在数据库中打开了一个cursor。
但是在执行完一次结果集后,并没有调用prepareStatementd 的close()方法即:
没有关闭游标 从而超出了ORACLE设置的最大值。

要解决这个问题需要修改代码,请下需求改代码。

以下是网上找的解决办法


方案1:
*********************************************************


很多朋友在Java开发中,使用Oracle数据库的时候,经常会碰到有ORA-01000: maximum opencursors
     exceeded.的错误。
       实际上,这个错误的原因,主要还是代码问题引起的。
       ora-01000: maximum open cursors exceeded.
       表示已经达到一个进程打开的最大游标数。
       这样的错误很容易出现在Java代码中的主要原因是:Java代码在执行conn.createStatement()和conn.prepareStatement()的时候,实际上都是相当与在数据库中打开了一个cursor。尤其是,如果你的createStatement和prepareStatement是在一个循环里面的话,就会非常容易出现这个问题。因为游标一直在不停的打开,而且没有关闭。
       一般来说,我们在写Java代码的时候,createStatement和prepareStatement都应该要放在循环外面,而且使用了这些Statment后,及时关闭。最好是在执行了一次executeQuery、executeUpdate等之后,如果不需要使用结果集(ResultSet)的数据,就马上将Statment关闭。
       对于出现ORA-01000错误这种情况,单纯的加大open_cursors并不是好办法,那只是治标不治本。实际上,代码中的隐患并没有解除。
       而且,绝大部分情况下,open_cursors只需要设置一个比较小的值,就足够使用了,除非有非常特别的要求。
       ---
       如果你不使用连接池,那么就没有什么问题,一旦Connection关闭,数据库物理连接就被释放,所有相关Java资源也可以被GC回收了。
       但是如果你使用连接池,那么请注意,Connection关闭并不是物理关闭,只是归还连接池,所以PreparedStatement和ResultSet都被持有,并且实际占用相关的数据库的游标资源,在这种情况下,只要长期运行,往往就会报“游标超出数据库允许的最大值”的错误,导致程序无法正常访问数据库。

       ---
       这个关不关和使用不使用conn
     pool没有关系,一般操作是会是这样,线程从外界获取一个conn,然后创建自己地stmt,rs,然后执行逻辑操作,然后将conn返回给pool。
     如果程序员忘记手动关地话。当这个线程执行完以后,stmt,rs都成垃圾,当他们被垃圾搜集地时候,gc会替我们把它们给关闭地。这就是很多代码没有关闭,仍然正常运行。

       但是这样会有一个潜在地问题。就是gc无法确定什么时候运行。如果free地内存很多,很可能有些gc就不会被启动,这样stmt迟迟没有被关闭,执行一段时间会报错。

       所以健壮地代码应该手工把rs,stmt都关闭
       ---
       Java连结Oracle常犯错误
       1。只懂 createStatement,不懂关闭statement
       2.。只懂 createStatement,不懂preparedStatement.
       3 。只懂在sql里用to_date,甚至直接用String,不懂用 setDate()
       ---
       我记得.我的程序中也出现过这种问题,
       主要原因是我get Connection 对象后,这个connectin没有被进行关闭,
       同时进行出来的 preparedStatement 对象,不关闭也会出现这种问题,
       而且,推荐这些数据库操作的变量尽量用局部变量,现用现取,随时关闭,而且放在finally{}中进行关闭,比较保险
       ---
       通常這樣的情形是你使用了超過DB設定同時可用的connection數量
       你可以試試看下列方式:
       1. 若你使用connection pool, 你可以將connection pool的max connection
       數量降低看看, 同時檢查有無歸還connection
       2. 若未使用conneciton pool, 你就該檢查一下, 你的connection在使用過後
       有沒有關閉囉
       另外一種情形是: 你的系統使用量真的很大, 所以同時150的DB connection
       session是不夠的, 你就需要調整Oracle DB囉
     很多朋友在Java开发中,使用Oracle数据库的时候,经常会碰到有ORA-01000: maximum opencursors
     exceeded.的错误。实际上,这个错误的原因,主要还是代码问题引起的。 ora-01000: maximum opencursors
     exceeded. 表示已经达到一个进程打开的最大游标数。
     这样的错误很容易出现在Java代码中的主要原因是:Java代码在执行conn.createStatement()和conn.prepareStatement()的时候,实际上都是相当与在数据库中打开了一个cursor。尤其是,如果你的createStatement和prepareStatement是在一个循环里面的话,就会非常容易出现这个问题。因为游标一直在不停的打开,而且没有关闭。一般来说,我们在写Java代码的时候,createStatement和prepareStatement都应该要放在循环外面,而且使用了这些Statment后,及时关闭。最好是在执行了一次executeQuery、executeUpdate等之后,如果不需要使用结果集(ResultSet)的数据,就马上将Statment关闭。对于出现ORA-01000错误这种情况,单纯的加大open_cursors并不是好办法,那只是治标不治本。实际上,代码中的隐患并没有解除。
     而且,绝大部分情况下,open_cursors只需要设置一个比较小的值,就足够使用了,除非有非常特别的要求。 ---
     如果你不使用连接池,那么就没有什么问题,一旦Connection关闭,数据库物理连接就被释放,所有相关Java资源也可以被GC回收了。
     但是如果你使用连接池,那么请注意,Connection关闭并不是物理关闭,只是归还连接池,所以PreparedStatement和ResultSet都被持有,并且实际占用相关的数据库的游标资源,在这种情况下,只要长期运行,往往就会报“游标超出数据库允许的最大值”的错误,导致程序无法正常访问数据库。
     --- 这个关不关和使用不使用conn
     pool没有关系,一般操作是会是这样,线程从外界获取一个conn,然后创建自己地stmt,rs,然后执行逻辑操作,然后将conn返回给pool。
     如果程序员忘记手动关地话。当这个线程执行完以后,stmt,rs都成垃圾,当他们被垃圾搜集地时候,gc会替我们把它们给关闭地。这就是很多代码没有关闭,仍然正常运行。
     但是这样会有一个潜在地问题。就是gc无法确定什么时候运行。如果free地内存很多,很可能有些gc就不会被启动,这样stmt迟迟没有被关闭,执行一段时间会报错。
     所以健壮地代码应该手工把rs,stmt都关闭 --- Java连结Oracle常犯错误 1。只懂
     createStatement,不懂关闭statement 2.。只懂createStatement,不懂preparedStatement. 3
     。只懂在sql里用to_date,甚至直接用String,不懂用 setDate() ---我记得.我的程序中也出现过这种问题,
     主要原因是我get Connection 对象后,这个connectin没有被进行关闭, 同时进行出来的preparedStatement
     对象,不关闭也会出现这种问题,而且,推荐这些数据库操作的变量尽量用局部变量,现用现取,随时关闭,而且放在finally{}中进行关闭,比较保险
     --- 通常這樣的情形是你使用了超過DB設定同時可用的connection數量 你可以試試看下列方式: 1.若你使用connection
     pool, 你可以將connection pool的max connection 數量降低看看, 同時檢查有無歸還connection2.
     若未使用conneciton pool, 你就該檢查一下, 你的connection在使用過後 有沒有關閉囉另外一種情形是:
     你的系統使用量真的很大, 所以同時150的DB connection session是不夠的, 你就需要調整OracleDB囉

 

 

 

方案2:
*********************************************************
[TIP]关于Java开发中使用Oracle数据库的一点注意事项(原创)

很多朋友在Java开发中,使用Oracle数据库的时候,经常会碰到有ORA-01000: maximum opencursors exceeded.的错误。

实际上,这个错误的原因,主要还是代码问题引起的。
ora-01000: maximum open cursors exceeded.
表示已经达到一个进程打开的最大游标数。

这样的错误很容易出现在Java代码中的主要原因是:Java代码在执行conn.createStatement()和conn.prepareStatement()的时候,实际上都是相当与在数据库中打开了一个cursor。尤其是,如果你的createStatement和prepareStatement是在一个循环里面的话,就会非常容易出现这个问题。因为游标一直在不停的打开,而且没有关闭。

一般来说,我们在写Java代码的时候,createStatement和prepareStatement都应该要放在循环外面,而且使用了这些Statment后,及时关闭。最好是在执行了一次executeQuery、executeUpdate等之后,如果不需要使用结果集(ResultSet)的数据,就马上将Statment关闭。

对于出现ORA-01000错误这种情况,单纯的加大open_cursors并不是好办法,那只是治标不治本。实际上,代码中的隐患并没有解除。
而且,绝大部分情况下,open_cursors只需要设置一个比较小的值,就足够使用了,除非有非常特别的要求。

我昨天犯的错误就是把创建statement放到了循环之内,结果导致cursor猛增,到达了数据库设置的最大数。
虽然解决了问题,也清楚了游标是何时创建的,但还有几个问题比较迷糊。
1、通过查询,JDBC是将所有的查询结果一次性放到ResultSet中,还是一次只放一定数目的记录?例如,查询结果为2000条数据,JDBC是一次性将2000条数据放到结果集中,还是分批放置呢?

2、当通过ResultSet.next(),移动结果集指针时,此时是否还与数据库发生交互?


方案3:
*********************************************************

ORA-01000:maximumopencursorsexceeded
2009-03-28 20:16:43   来源:网络搜集浏览次数: 网友评论 条
      错误的中文解释为:游标超出最大连接数
  游标具体是个什么东西就不多说了,只说明一点:游标有显式游标和隐式游标,并不是没有显式的调用游标就不会使用到游标,有些SQL语句是会调用隐式游标的,这就是为什么有人没有显式的调用游标却出现了上面的错误提示。
  出现这种错误提示大多是因为OracleConnection,OracleDataReader使用完后没有正常的关闭造成的。只要使用后立即关闭掉就可以了,即调用相应的Close()就可以了。对于OracleDataReader来说一定要调用Close()关掉。
  用以上方法就可以解决问题,下面是我解决问题的过程,毕竟“授之以鱼,不如授之以渔”
  OracleDataReader比较特殊,使用完后如果没有关掉,只关掉了OracleConnection,Oracle数据库中的游标还是会存在的。当系统运行一段时间,未关掉的游标会越来越多,当超出ORACLE设置的最大值时也就出现了上诉的错误。这个最大值以ORACLE10g为例是300,保存在oracle\product\10.1.0\admin\orcl\pfile目录下的init.ora文件的open-cursors参数里。
  首先对想通过吧open—sursors这值改大些来避免问题的人说,这样改的话有可能会避免该问题,使其以后不再出现,但是避免问题不是解决问题的好办法,因为这样处理问题会留下很大的隐患,问题不知道什么时候又会出现。到时候在哭就来不及了。要发现问题的产生原因,然后解决它。
  以下是发现问题的方法。
  select * from v_$open_cursor where user_name='****'
  以sys用户登录后,执行以上语句试下能看到什么,其中那个****是程序中连接数据库的用户名。执行后可以看到这是查看****用户未释放的游标是有那些SQL语句执行的。是不是很绕口的一句话。动手试下你就会明白了。
  如果吧SQL语句改成 select count(*) from v_$open_cursor whereuser_name='****'又会怎样呢?那就是这个用户当前有几个游标未释放。看这个值有什么意义呢?前面提到过ORACLE10G的默认最大游标数是300吧。对了。当查看的这个值超过300就出现问题了。你可以找个调用了OracleDataReader的地方,在执行代码前后分别执行下这个SQL语句,你就明白游标数为什么会一致增大,最后导致超出最大游标数了。
  如果在遇到ORACLE游标的问题不妨用这种方法找找产生这些游标的地方到底在哪里。愿意的话也可以用OracleDataReader执行不同的SQL语句,看看那些SQL语句会调用到隐式游标。
  文章些的有点乱,各位将就的看看吧。哈哈。希望会有什么帮助。有什么问题的话记得回帖啊。我们交流交流。

原创粉丝点击