次のコードは、Java 6 (およびそれ以前) では問題なく動作していましたが、JRE 7 (Java 7) に更新すると動作しなくなりました。
URL は FTP ファイルです。
ftp://ftp-private.ncbi.nlm.nih.gov/pubchem/.fetch/96/4133257873201306969.sdf.gz
これが私が得る出力です:
application/octet-stream -1 [Ljava.lang.StackTraceElement;@5419f97c]
そして、ここに私のコードがあります:
public static void store(URL url, File targetFile){
try
{
System.out.println(url);
URLConnection uc = url.openConnection();
String contentType = uc.getContentType();
System.out.println(contentType);
int contentLength = uc.getContentLength();
System.out.println(contentLength);
Settings.setDownloadSize(contentLength);
if (contentType.startsWith("text/") || contentLength == -1) {
throw new IOException("This is not a binary file.");
}
InputStream raw = uc.getInputStream();
InputStream in = new BufferedInputStream(raw);
byte[] data = new byte[contentLength];
int bytesRead = 0;
StatusPanel.updateProgrssBar(bytesRead);
int offset = 0;
while (offset < contentLength) {
bytesRead = in.read(data, offset, data.length - offset);
if (bytesRead == -1) {
break;
}
offset += bytesRead;
StatusPanel.updateProgrssBar(offset);
}
in.close();
if (offset != contentLength) {
throw new IOException("Only read " + offset + " bytes; Expected " + contentLength + " bytes");
}
FileOutputStream out = new FileOutputStream(targetFile);
out.write(data);
out.flush();
out.close();
//StatusPanel.setStatus("File has been stored at " + targetFile.toString());
//System.out.println("file has been stored at " + targetFile.toString());
}
コンテンツの長さは -1 を返します。
Area: API: Networking
Synopsis: Server Connection Shuts Down when Attempting to Read Data When http Response Code is -1
このコードを Java 7 と互換性を持たせるにはどうすればよいですか?
説明: CR 6886436 のバグ修正の結果、HTTP プロトコル ハンドラは、有効な HTTP ステータス行なしで応答を送信するサーバーへの接続を閉じます。これが発生すると、その接続でデータを読み取ろうとすると、IOException が発生します。
たとえば、次のコードは問題があります。
public static void test () throws Exception {
.....
HttpURLConnection urlc = (HttpURLConnection)url.openConnection();
....
System.out.println ("Response code: " + urlc.getResponseCode());
/** Following line throws java.io.IOException: Invalid Http response
* when Response Code returned was -1
*/
InputStream is = urlc.getInputStream(); // PROBLEMATIC CODE
この問題を回避するには、getResponseCode メソッドからの戻り値を確認し、-1 値を適切に処理します。おそらく、新しい接続を開くか、ストリームで getErrorStream を呼び出します。非互換性の性質: 行動 RFE: 7055058
問題は間違いなくgetContentLength()
方法にあります。
JRE6 ではこのメソッドは値を返しますが、JRE7 では -1 になります。