
メモリリーク(?)時にSQLException
初めて質問させていただきます。
java+mysqlでWebアプリを作成しているのですが、DBからデータ取得後、取得結果をStringに格納するところで2回に1回だけエラーとなってしまいます。
具体的には下記のようなプログラム実行時にSQLExceptionが出てしまいます。
// ---- プログラム
// ・・・(略)・・・
String str = "";
rs = pst.executeQuery();
while (rs.next())}
str = str + rs.getString("name");
}
// ---- 出力されるエラー(途中まで)
java.sql.SQLException: Operation not allowed after ResultSet closed
at com.mysql.jdbc.SQLError.createSQLException(SQLError.java:910)
at com.mysql.jdbc.ResultSet.checkClosed(ResultSet.java:644)
at com.mysql.jdbc.ResultSet.next(ResultSet.java:6663)
ResultSetがクローズされてるよというエラーだと思うのですが、
str = str + rs.getString("name");
のところを
str = rs.getString("name");
に書き換えた場合と、String型ではなくStringBufferでappendを使って文字列連結させていった場合はエラーになりません。
また、ログを入れて動作状況を確認してみたところwhile文は何回かは動いているようです。
※while文は5000回以上ループし、strは最終的にかなり大きいサイズになります。
このような状況を考えると、メモリリークが発生して落ちているのかと思うのですが、発生しているエラー内容は上記の通りSQLExceptionです。
また、一番腑に落ちないのはこのエラーが2回に1回だけ出る、ということです。
同じような現象をご存知の方がいらっしゃいましたら、何が起こっているのか教えていただけないでしょうか?
未熟ながら私の予想ですと、メモリリークが発生する前にjavaがGCでメモリクリアしようとしているんだけど、そのときにResultSetも一緒に消しちゃってる・・・?
javaやjdbcのバグ???とも考えているのですが、どうやって確認したらいいのかわからず困っています。
No.2ベストアンサー
- 回答日時:
java のgcがResultSetを消しちゃうと言う様な話は、聞いた事がありませんが、
ResultSet自体が、時間切れや、過負荷時に、自動的にクローズされると言う
事なら、ありえるかと思います。
ResultSet自体の実装として、TCP/IP等による、ソケット接続を内部的に行っ
ている可能性がありますからねぇ。
>ResultSet自体が、時間切れや、過負荷時に、自動的にクローズされると言う
>事なら、ありえるかと思います。
なるほど。
1/2の確立でしかエラー発生しないことを考えると可能性としては低い気もしますが、その線で調べてみます。ありがとうございます!
No.5
- 回答日時:
> 変数rsは使う前に毎回初期化していますので、問題ないかと思います。
どう初期化しているのかわからないけど
「~思います」がついているうちは疑ってかかるわよ。
私はResultSetの使い方を疑っているわ。
closeされた後のResultSetに対し何かやっているんじゃないか、ってね。
そのひとつが変数rsの再利用だったわけ。
もう一度、なにか思い込みでやっているところがないか
見直してみるべきよ。
No.4
- 回答日時:
PreparedStatementを使い回ししている可能性はありませんか?
参考URLより:
デフォルトでは、Statement オブジェクトごとに 1 つの ResultSet オブジェクトだけが同時にオープンできます。
rsやpstがスレッドセーフでは無いのに、他のスレッドから(他のリクエストから)サーブレットが呼ばれていませんか?
参考URL:http://java.sun.com/javase/ja/6/docs/ja/api/java …

No.1
- 回答日時:
>while (rs.next())
rsはResultSetクラスですかね。java.sql のクラスは使ったことがないけど、
Iteratorを実装しているなら、次が有るかどうかは、hasnext() でチェックして、
次があれば、ifブロック内で next() でobject を得て作業という手順のはず。
次があるかのチェックもしないで、next() を繰り返せば、当然いつかは、「データを得られなかった → Operation not allowed after ResultSet closed」というエラーになる。
この回答への補足
回答ありがとうございます。
また、説明不足で申し訳ありません。ご指摘の通り、rsはResultSetです。
しかし、ResultSetはnext()で次が無かったらfalseを返すので、
>while (rs.next())
は問題ないかと思います。
ご指摘にあるように「いつか必ずエラーになる」のではなく、2回に1回のみエラーになっています。
お探しのQ&Aが見つからない時は、教えて!gooで質問しましょう!
似たような質問が見つかりました
- オープンソース Python openpyxlを使用したセル番地の使用について 1 2023/08/03 22:05
- C言語・C++・C# 宣言する関数の形が決まっている状態で、 str1とstr2の文字列をこの順に引っ付けてstrに保存し 2 2022/05/30 18:21
- Excel(エクセル) PHPプログラムをエクセルに張り付けると検索ボックスがでてくる! 3 2022/05/08 07:10
- Visual Basic(VBA) シートをコピーする下記記述でダイアログを用いた記述がわかりません?( A = Dir(ThisWor 4 2022/08/22 12:26
- C言語・C++・C# c言語 プログラムのエラー 1 2023/02/11 20:31
- C言語・C++・C# C#テキストボックスの文字を配列にいれてその後表示する 4 2022/07/17 04:47
- C言語・C++・C# str[j++]の意味 2 2022/08/30 16:20
- C言語・C++・C# プログラミングの授業の課題です 1 2023/01/17 22:15
- C言語・C++・C# sprintf()の使い方について 1 2022/08/17 16:16
- Visual Basic(VBA) VBAでfunctionを利用しようとしたときに「引数は省略できません」というエラーが出ます 1 2022/10/15 16:30
関連するカテゴリからQ&Aを探す
おすすめ情報
デイリーランキングこのカテゴリの人気デイリーQ&Aランキング
-
Eclipseで、プロジェクト名のと...
-
JDK1.3のエラー表示について
-
Buttonを配列とすることができ...
-
JSP内でString型からint型に変...
-
コンパイル時にエラーが出てし...
-
JSP/サーブレットで一般的な入...
-
eclipseでのsvn認証エラー回避方法
-
javaもしくはjava3Dをつかって
-
JAVAコンパイルエラー全て...
-
エラー:org.springframeworkが...
-
tomcat起動時エラー
-
シンボルが見つかりませんとい...
-
JSPで意味不明のコンパイルエラー
-
タグが閉じてないというエラー...
-
MySQLを利用したサーブレットの...
-
getParameterについて
-
構文エラーのやり直しについて
-
拡張子 class ファイルを開きた...
-
Javaのエラーについてです
-
eclipseでのライブラリの追加に...
マンスリーランキングこのカテゴリの人気マンスリーQ&Aランキング
-
Eclipseで、プロジェクト名のと...
-
エラー:org.springframeworkが...
-
コンパイル時にエラーが出てし...
-
eclipseをバージョンアップした...
-
tomcat起動時エラー
-
構文エラーのやり直しについて
-
タグが閉じてないというエラー...
-
Eclipse デバッグ エラー処理
-
Javaのエラーについてです
-
eclipseでのsvn認証エラー回避方法
-
Eclipseの環境について
-
シンボルが見つかりませんとい...
-
java eclipse apache tomcat 9....
-
java エラー
-
【ジェネリックス】コンパイル...
-
eclipseのエラーについて質問で...
-
BufferdReaderが見つからない?
-
javamailにて連続送信の際のエラー
-
拡張子 class ファイルを開きた...
-
JSP NullPointerException
おすすめ情報