Использование Apache HttpClient как установить тайм-аут для запроса и ответа

мне нужно установить время ожидания для Http-запроса, который мы делаем для службы (не веб-службы). Мы используем HTTP-клиент Apache. Я добавил Эти 2 строки кода, чтобы установить тайм-аут на запрос и ответ на услугу.

HttpConnectionParams.setConnectionTimeout(params, 10000);
HttpConnectionParams.setSoTimeout(params, 10000);

1) в настоящее время я установил 10 секунд в качестве таймаута, так как я вижу ответ, поступающий от службы почти мгновенно. Увеличить или уменьшить время?

2) что произойдет, когда ответ займет больше 10 секунд? Будет ли это исключение и какое исключение будет? Есть ли что-то еще, что мне нужно добавить, чтобы установить время в приведенном ниже коде.

public HashMap<String, Object> getJSONData(String url) throw Exception{
    DefaultHttpClient httpClient = new DefaultHttpClient();
    HttpParams params = httpClient.getParams();
    HttpConnectionParams.setConnectionTimeout(params, 10000);
    HttpConnectionParams.setSoTimeout(params, 10000);
    HttpHost proxy = new HttpHost(getProxy(), getProxyPort());
    ConnRouteParams.setDefaultProxy(params, proxy);
    URI uri;
    InputStream data = null;
    try {
        uri = new URI(url);
        HttpGet method = new HttpGet(uri);
        HttpResponse response = httpClient.execute(method);
        data = response.getEntity().getContent();
    }
    catch (Exception e) {
        e.printStackTrace();
    }
    Reader r = new InputStreamReader(data);
    HashMap<String, Object> jsonObj = (HashMap<String, Object>) GenericJSONUtil.fromJson(r);
    return jsonObj;
}

2 ответов


исключения, которые вы увидите, будут ConnectTimeoutException и SocketTimeoutException. Фактические значения таймаута, которые вы используете, должны быть максимальным временем ожидания вашего приложения. Одно важное замечание о таймауте чтения заключается в том, что он соответствует таймауту чтения сокета. Таким образом, это не время, разрешенное для полного ответа, а скорее время, предоставленное для чтения одного сокета. Таким образом, если есть 4 чтения сокетов, каждый из которых занимает 9 секунд, общее время чтения составляет 9 * 4 = 36 секунд.

если вы хотите указать общее время для ответа (включая подключение и общее время чтения), вы можете обернуть вызов в поток и использовать тайм-аут потока для этого. Например, я обычно делаю что-то вроде этого:

Future<T> future = null;
future = pool.submit(new Callable<T>() {
    public T call() {
        return executeImpl(url);
    }   
}); 

try {
    return future.get(10, TimeUnit.SECONDS);
}   
catch (InterruptedException e) {
    log.warn("task interrupted", name);
}   
catch (ExecutionException e) {
    log.error(name + " execution exception", e); 
}   
catch (TimeoutException e) {
    log.debug("future timed out", name);
}

некоторые предположения, сделанные в коде выше: 1) это в функции с параметром url, 2) это в классе с переменной имени, 3) log является экземпляром log4j, и 4) пул является исполнителем некоторого пула потоков. Обратите внимание, что даже если вы используете тайм-аут потока следует также указать время ожидания подключения и сокета на HttpClient, чтобы медленные запросы не съедали ресурсы в пуле потоков. Также обратите внимание, что я использую пул потоков, потому что обычно я использую это в веб-службе, поэтому пул потоков совместно используется в группе потоков tomcat. Ваша среда может отличаться, и вы можете просто создать новый поток для каждого вызова.

кроме того, я обычно вижу тайм-ауты, установленные через функции-члены params, например это:

params.setConnectionTimeout(10000);
params.setSoTimeout(10000);

но, возможно, ваш синтаксис также работает (не уверен).


Я предполагаю, что многие люди приходят сюда из-за названия и потому, что HttpConnectionParams API устарел.

используя последнюю версию HTTP-клиента Apache, вы можете установить эти таймауты с помощью параметров запроса:

HttpPost request = new HttpPost(url);

RequestConfig requestConfig = RequestConfig.custom()
  .setSocketTimeout(TIMEOUT_MILLIS)
  .setConnectTimeout(TIMEOUT_MILLIS)
  .setConnectionRequestTimeout(TIMEOUT_MILLIS)
  .build();

request.setConfig(requestConfig);

кроме того, вы также можете установить это при создании HTTP-клиента, используя API builder для HTTP-клиента, но вам также нужно будет создать пользовательский диспетчер соединений с пользовательской конфигурацией сокета.

на настройки пример файла является отличным ресурсом, чтобы узнать о том, как настроить Apache HTTP Client.