web-dev-qa-db-ja.com

HttpURLConnection Java.io.FileNotFoundException

以下のコードは、Apacheサーバーのように見えるものに接続するとうまく機能しますが、.Netサーバーに接続しようとするとエラーがスローされます。私はそれがヘッダーの要件であると推測していますが、何を試しても成功した応答を得ることができないようです。

public String Download(String Url)
{
 String filepath=null;
 try {
  //set the download URL, a url that points to a file on the internet
  //this is the file to be downloaded
  URL url = new URL(Url);
  //create the new connection
  HttpURLConnection urlConnection = (HttpURLConnection) url.openConnection();

  //set up some things on the connection
  urlConnection.setRequestMethod("GET");
  urlConnection.setDoOutput(true); 
   //and connect!
  urlConnection.connect();
  //set the path where we want to save the file
  //in this case, going to save it on the root directory of the sd card.
  File SDCardRoot = Environment.getExternalStorageDirectory();
  //create a new file, specifying the path, and the filename
  //which we want to save the file as.

  String filename= "effortback.png";   // you can download to any type of file ex:.jpeg (image) ,.txt(text file),.mp3 (audio file)
  Log.i("Local filename:",""+filename);
  File file = new File(SDCardRoot + "/",filename);

  //=====================================
  if(file.createNewFile())
  {
   file.createNewFile();
  }
  //=====================================

  //this will be used to write the downloaded data into the file we created
  FileOutputStream fileOutput = new FileOutputStream(file);

  //this will be used in reading the data from the internet
  InputStream inputStream = urlConnection.getInputStream();

  //=====================================
  //this is the total size of the file
  int totalSize = urlConnection.getContentLength();
  //variable to store total downloaded bytes
  int downloadedSize = 0;
  //=====================================

  //create a buffer...
  byte[] buffer = new byte[2048];
  int bufferLength = 0; //used to store a temporary size of the buffer

  //now, read through the input buffer and write the contents to the file
  while ( (bufferLength = inputStream.read(buffer)) > 0 ) {
   //add the data in the buffer to the file in the file output stream (the file on the sd card
   fileOutput.write(buffer, 0, bufferLength);
   //add up the size so we know how much is downloaded
   downloadedSize += bufferLength;
   //this is where you would do something to report the prgress, like this maybe
   Log.i("Progress:","downloadedSize:"+downloadedSize+"totalSize:"+ totalSize) ;

  }
  //close the output stream when done
  fileOutput.close();
  if(downloadedSize==totalSize)   filepath=file.getPath();

 //catch some possible errors...
 } catch (MalformedURLException e) {
  e.printStackTrace();
  Log.i("URL-ERROR:",e.toString());
 } catch (IOException e) {
  filepath=null;
  e.printStackTrace();
  Log.i("IO-ERROR:",e.toString());
 }
 Log.i("filepath:"," "+filepath) ;

 return filepath;

}

エラーの範囲:

Java.io.FileNotFoundException  //I always get this with either one of the below
   org.Apache.harmony.luni.internal.net.www.protocol.http.HttpURLConnection.getInputStream(HttpURLConnection.Java:1061)

//or this one below
libcore.net.http.HttpURLConnectionImpl.getInputStream(HttpURLConnectionTmpl.Java:186)

私が何を試みても、私はそれを動作させることができないようです。繰り返しますが、Googleや他のサイトから画像をダウンロードしようとしても機能しますが、すべてではなく、間違いなく私の(.Net)ではありません。ここで何が欠けていますか?助けてください。

42
nathan

サーバーが> =HTTPStatus.BAD_REQUEST(400)を返す場合、FileNotFoundException(およびHttpUrlConnection)からOkHttpClientを取得できます。最初にステータスコードを確認して、読み取る必要のあるストリームを確認する必要があります。

int status = connection.getResponseCode();

if(status >= HttpStatus.SC_BAD_REQUEST)
    in = connection.getErrorStream();
else
    in = connection.getInputStream();

HttpStatusは非推奨です。最新の構文は次のようです:

            InputStream inputStream;

            int status = urlConnection.getResponseCode();

            if (status != HttpURLConnection.HTTP_OK)
                inputStream = urlConnection.getErrorStream();
            else
                inputStream = urlConnection.getInputStream();
101
Dori

あなたが言ったように同じ問題を解決しました:

_urlConnection.setDoOutput(false);
_

デフォルトでtrueであるため、falseに設定する必要があることに注意してください。
_VolleyHurlStackがtrueに設定していたため、falseに設定する必要があることに注意してください。
気を付けて ;)

編集:コードをチェックしたところ、デフォルトでは@Abraham Philipが言ったようにfalseです。ただし、getDoOutput()を呼び出した場合はtrueを返すため、falseに設定する必要がありました。 VolleyHurlStackがtrueに設定していることがわかりました。したがって、私の結論は、setDoOutputをfalseにすればうまくいきます。

_package Java.net;
public abstract class URLConnection {
...

    /**
     * Specifies whether this {@code URLConnection} allows sending data.
     */
    protected boolean doOutput;

...
    public boolean getDoOutput() {
        return doOutput;
    }

    public void setDoOutput(boolean newValue) {
        checkNotConnected();
        this.doOutput = newValue;
    }
_
48
GuilhE

HEAD- Requestを実行するとFileNotFoundExceptionがスローされるという問題に直面しました。
その理由は、HEADへの応答に本文がないため、getInputStreamを呼び出すとFileNotFoundExceptionがスローされるためです。

したがって、HEADを実行している場合、InputStreamを取得しようとしないでください。

0
Springrbua