2012年11月26日 星期一

Microsoft SQL Server Driver for JDBC 發生 ResultSet can not re-read row data for column X 解决方法


問題發生的經過是,某天資料表須新增欄位,因為不確定長度要開多少,所以就先開
varchar(50),後來不夠又改成varchar(max)接著惡夢開使,所有服務一個接一個掛,讀資料庫接丟出以下例外訊息

Java.sql.SQLException: [Microsoft] [SQL Server 2000 JDBC Driver] ResultSet can not re-read row data for column X ( X = 1 ~ N )

Google了一下網路上的文章,才發現原來是JDBC驅動程式的問題。
微軟官方所提供的說明網頁:http://support.microsoft.com/kb/824106
第一種是,取資料時必須由資料表(Table)的最左邊一欄開始,
依序向右,例如我有id, name, passwd, comments等4個欄位,取資料的順序就只能:

rs.getString("name");
rs.getString("passwd");
rs.getString("comments");

而不能是:
rs.getString("name");
rs.getString("id");
rs.getString("passwd");
rs.getString("comments");

第二種是,此外同一筆資料的每個欄位,也只能被讀取一次,例如:
rs.getString("name");
rs.getString("name");
上述情況讀取了2次,就會出現錯誤。

處理方式可以用宣告變數存放:
String strId = rs.getString("id");
String strName = rs.getString("name");

可是我發現古人的程式都有以上這兩個問題,公司服務又2, 30個,改不完阿....冏
繼續Google

(1)如果采用jdbc-odbc驱动,那么就必须按照查询顺序来一次读取(不论有没有image或text类型)
(2)如果采用微软提供的ms sql server jdbc driver,如果查询语句中,不存在image或text类型字段,那么可以按照无序获取
(3)如果采用微软提供的ms sql server jdbc driver,如果查询语句中,存在image或text类型字段,那么就必须按照顺序读取,否则就会报告Driver]ResultSet can not re-read row data for column之类的错误
(4)如果想不查询语句中有没有image或text类型字段,都可以不按照顺序获取,或重复获取。

當我努力回想我到底改了什麼之後,才發現"如果查询语句中,存在image或text类型字段"
直覺覺的是我把varchar(50)改成varchar(max)關係,果然改回來就正常了.....冏到爆

網路上有人直接是建議改用JTDS


參考來源
http://www.wretch.cc/blog/weiyushen/9979951
http://www.ourys.com/post/ResultSet_can_not_re-read_row_data_for_column.html
http://android.blog.51cto.com/268543/49995