天道酬勤

没有成功,说明努力还不够!
随笔 - 2, 文章 - 0, 评论 - 29, 引用 - 0
数据加载中……

关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)

昨天在开发过程中,经常碰到ORA-01000: maximum open cursors exceeded这个错误。一看报错,知道是游标超过最大数了,以前在做DB2开发的时候,也曾遇到过类似的问题。虽然解决了,但一直没弄清楚,JDBC是如何和数据库发生交互的?又是在什么时候打开数据库游标的?脑袋就跟浆糊一样,说来真是惭愧啊

昨天google了一把,终于解开一点迷惑了。

下面的片断转载于www.itpub.net

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

很多朋友在Java开发中,使用Oracle数据库的时候,经常会碰到有ORA-01000: maximum open cursors 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(),移动结果集指针时,此时是否还与数据库发生交互?

posted on 2005-01-27 10:15 候鸟南飞 阅读(52569) 评论(27)  编辑  收藏

评论

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)  回复  更多评论   

连接和statement使用完毕之后应该及时关闭,这个应该是粉简单的道理吧.
2005-01-27 10:53 | eamoi

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)  回复  更多评论   

鸟人
还没回到阿?
2005-02-17 21:33 | 不至于

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)  回复  更多评论   


2005-03-29 15:05 |

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)  回复  更多评论   

The article is great. I got the same error and keep scratching my head till I found your page. Keep it up!
2005-04-28 15:39 | San

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)  回复  更多评论   

继续下去啊。这个问题应该是一个比较典型的问题了吧,继续讨论阿
2005-05-10 18:27 | 芨芨草

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)  回复  更多评论   

pst.close();
rs.close();

只要及时关闭应该就可以了吧?
2005-07-05 15:23 | plutohades

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)  回复  更多评论   

首先结果集resultset在你的java程序处。其中有个fetchsize设置,这个表示每次从数据库处取多少条记录到resultset.
2006-03-14 09:42 |

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)  回复  更多评论   

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

不发生交互,数据库执行完查询后,已经把查询结果交给ResultSet了,以后的操作,和数据库无关。
2006-03-15 10:26 | jing

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)  回复  更多评论   

交互,如果1个查询有10000条记录,resultset中只有fetchsize条,当next时还会在一定时机去交互
2006-03-17 10:44 |

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)  回复  更多评论   

白是对的。
2006-05-08 09:27 | 阿黑

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)  回复  更多评论   

非常感谢!
2006-05-12 15:00 | 感谢

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)  回复  更多评论   

晕。。。。。。。。。
2006-05-18 15:50 | 顺哥

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)  回复  更多评论   

一搜索这个问题竟然到了你的blog,真是冤孽啊。。:)
2006-05-18 15:51 | 顺哥

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)  回复  更多评论   

不是将查询结果一次性放到ResultSet中, 而是分批放入ResultSet中,一般情况下是每次10条记录.
当通过ResultSet.next(),移动结果集指针时,此时还会与数据库发生交互.

导致maximum open cursors exceeded这个问题一般是代码本身的问题.
比如执行一个查询没有关闭游标,或是在存储过程中打开了游标而没有关闭.
特别是出现异常情况是没有关闭游标.严重时会导致系统崩掉
2006-06-01 03:58 | sheen

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)  回复  更多评论   

我也是同样的问题,但我开发语言是用DELPHI,使用组件是ADOQUERY,游标超过最大数困扰我好久了,如果有谁知道,请告之jxprovince@mail.china.com
2006-06-18 17:42 | JXPROVINCE

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)  回复  更多评论   

啊~!~`我刚也犯了这种错误!!``看到这里的贴子,小妹又明白了:"哦!``原来conn.createStatement()和conn.prepareStatement()的时候实际上都是相当与在数据库中打开了一个cursor。!!"明白了这点过后,我就在循环里新建一个prepareStatement对象,操作完一次,成功过后就马上关掉!这样,循环里每次一个insert都新new 一个prepareStatement 操作成功完成之后再马上关闭.嘎嘎.~这样,cursor就不会溢出拉.
2007-01-11 09:55 | 子圆

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)  回复  更多评论   

我遇到的就是这个问题,看了你的文章终于明白了
2007-03-07 18:20 | vingle

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)  回复  更多评论   

有时候出于性能方面的考虑,多个用户使用一个数据库连接,这时一定要及时关闭cursor,否则就会报这个错。
2007-06-15 11:15 | fenix

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)  回复  更多评论   

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


不与数据库交互,此时数据集拿到一个collection中,ResultSet就是一个collectin是实现
2007-07-11 18:00 | cshbl@hotmail.com

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)  回复  更多评论   

3ks
2007-10-11 18:06 | zhangyangbin

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)  回复  更多评论   

谢谢
犯了跟你一样的错
2008-09-01 10:03 | nijj

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)  回复  更多评论   

你是對的...
2008-12-16 18:47 | 豬頭

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)[未登录]  回复  更多评论   

我的问题还是这个,但是估计你没有回答到点子上。
我在开发过程中,本身用Hibernate3应该完全不用我们干预Session的工作吧,要不还要它作甚?
我的测试系统在MYSQL数据库上从来不会出现这个问题,但是切换到生产系统,除了数据库URL和方言配置改成ORACLE外,其他全部不变,但是它就是出了这个ORA-01000,这不是逼着我不用这个破Hibernate嘛
2009-06-02 08:10 | 明明

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)[未登录]  回复  更多评论   

好像是这样的,出现了同样的报错,看程序也是在循环里面prepareStatement
2009-08-10 10:41 | 111

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)  回复  更多评论   

ResultSet不是与数据库的一个映射关系吗,rs.close()之后,ResultSet.next()不行吧....
2010-08-12 11:40 | 任李菲

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)  回复  更多评论   

刚好碰到了这个问题,谢了、、、
2011-02-28 16:02 |

# re: 关于ORACLE游标的问题(ORA-01000: maximum open cursors exceeded)  回复  更多评论   

只要不循环或者过多的创建statement就行了,一般写循环逻辑会出现这类问题。
有些人提到jpa类的框架,这种情况如果出现不是框架的问题,而是使用方法不得当造成的,statement的类型选择或者sql的结构优化不对,导致statement创建过多。

next是否跟数据库交互,取决于你的fetchsize
2011-03-16 17:25 | 王志猛

只有注册用户登录后才能发表评论。


网站导航:
博客园   IT新闻   Chat2DB   C++博客   博问