Подтвердить что ты не робот

HttpURLConnection setConnectTimeout() не влияет

Я подключаюсь к простой RSS-ленте, используя HTTPUrlConnection. Он работает отлично. Я хотел бы добавить тайм-аут к соединению, так как не хочу, чтобы мое приложение зависало в случае плохого соединения или чего-то еще. Это код, который я использую, и метод setConnectTimeout не имеет никакого эффекта.

        HttpURLConnection http = (HttpURLConnection) mURL.openConnection();
        http.setConnectTimeout(15000); //timeout after 15 seconds
...

Если это помогает мне развиваться на андроиде.

4b9b3361

Ответ 1

Вы также должны попытаться установить тайм-аут чтения (http.setReadTimeout()). Зачастую веб-сервер с радостью принимает ваше соединение, но он может быть медленным в ответ на запрос.

Ответ 2

Вероятно, вы оба/оба:
1) Не читайте ничего из подключения
2) Не поймайте и не обработайте исключение должным образом

Как уже упоминалось здесь, используйте логику, подобную этой:

int TIMEOUT_VALUE = 1000;
try {
    URL testUrl = new URL("http://google.com");
    StringBuilder answer = new StringBuilder(100000);

    long start = System.nanoTime();

    URLConnection testConnection = testUrl.openConnection();
    testConnection.setConnectTimeout(TIMEOUT_VALUE);
    testConnection.setReadTimeout(TIMEOUT_VALUE);
    BufferedReader in = new BufferedReader(new InputStreamReader(testConnection.getInputStream()));
    String inputLine;

    while ((inputLine = in.readLine()) != null) {
        answer.append(inputLine);
        answer.append("\n");
    }
    in.close();

    long elapsed = System.nanoTime() - start;
    System.out.println("Elapsed (ms): " + elapsed / 1000000);
    System.out.println("Answer:");
    System.out.println(answer);
} catch (SocketTimeoutException e) {
    System.out.println("More than " + TIMEOUT_VALUE + " elapsed.");
}

Ответ 3

У меня была аналогичная проблема - потому что HttpUrlConnection не будет таймаутом в середине загрузки. Например, если вы отключите Wi-Fi при загрузке, мой продолжал говорить, что он загружается, застряв в том же проценте.

Я нашел решение, используя TimerTask, подключенный к AsyncTask с именем DownloaderTask. Попробуйте:

class Timeout extends TimerTask {
    private DownloaderTask _task;

    public Timeout(DownloaderTask task) {
        _task = task;
    }

    @Override
    public void run() {
        Log.w(TAG,"Timed out while downloading.");
        _task.cancel(false);
    }
};

Затем в цикле фактической загрузки установите таймер для ошибки таймаута:

                    _outFile.createNewFile();
                    FileOutputStream file = new FileOutputStream(_outFile);
                    out = new BufferedOutputStream(file);
                    byte[] data = new byte[1024];
                    int count;
                    _timer = new Timer();
                    // Read in chunks, much more efficient than byte by byte, lower cpu usage.
                    while((count = in.read(data, 0, 1024)) != -1 && !isCancelled()) { 
                        out.write(data,0,count);
                        downloaded+=count;
                        publishProgress((int) ((downloaded/ (float)contentLength)*100));
                        _timer.cancel();
                        _timer = new Timer();
                        _timer.schedule(new Timeout(this), 1000*20);
                    }
                    _timer.cancel();
                    out.flush();

Если он истечет, и не будет загружать даже 1K за 20 секунд, он отменяет, вместо того, чтобы появляться навсегда.

Ответ 4

Я столкнулся с той же проблемой. Установка connectionTimeout и readTimeout, похоже, не возвращает исключение, как ожидалось, но выполняйте действительно. Мне потребовалось время, чтобы проверить метод URLConnection() и понять, что происходит. В документации для setConnectTimeout есть предупреждение

", если имя хоста разрешено для нескольких IP-адресов, этот клиент будет пытаться каждый. Если подключение к каждому из этих адресов не удастся, произойдет несколько тайм-аутов до того, как попытка подключения выведет исключение." Это означает, что если у вас есть 10 ips, разрешенных вашим хостом, ваш фактический тайм-аут будет "10 * readTimeoutSet".

Вы можете проверить ips для имени хоста здесь

Ответ 5

http.setConnectTimeout(15000);
http.setReadTimeout(15000);

Ответ 6

Попробуйте установить ConnectionTimeout перед тем, как открыть соединение.