Использование 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.