2

我在我的android应用程序上使用restTemplate(来自spring android框架)来联系一个带有post的rest服务器,我使用一个AsyncTask来发送请求和两个类Request和Response,它们是以Json形式发送的POJO。

public class RequestSender extends AsyncTask<Object, Void, Response> {
    private RestTemplate restTemplate = new RestTemplate();
    private static final String SERVER_REQUEST_PATH = "/path/to/rest/service";

    @Override
    protected Response doInBackground(Object... args) {
        String url = (String) args[0] + SERVER_REQUEST_PATH;
        Request requestArgs = (Request) args[1];
            Response response = null;

        restTemplate.getMessageConverters().add(new MappingJackson2HttpMessageConverter());
        try {
               response = restTemplate.postForObject(url, requestArgs, Response.class);
        } catch (RestClientException e) {
            e.printStackTrace();
            response = null;
        }
        return response;
    }
}

如您所见,代码非常简单,并且大部分时间都可以正常工作,但不是按特定顺序:

  1. 我向服务器发出第一个(或更多)请求;
  2. 我重新启动服务器而不重新启动应用程序;
  3. 我向服务器发出新请求;

然后服务器重启后的第一个请求(步骤 3)不会通过 postForObject(...) 方法发送到服务器,但下一个请求是。这第一个请求给我一个错误:

07-10 10:24:13.402: W/System.err(4827): org.springframework.web.client.ResourceAccessException: I/O error: null; nested exception is java.io.EOFException
    07-10 10:24:13.402: W/System.err(4827):     at org.springframework.web.client.RestTemplate.doExecute(RestTemplate.java:491)
    07-10 10:24:13.402: W/System.err(4827):     at org.springframework.web.client.RestTemplate.execute(RestTemplate.java:439)
    07-10 10:24:13.402: W/System.err(4827):     at org.springframework.web.client.RestTemplate.postForObject(RestTemplate.java:317)
    07-10 10:24:13.402: W/System.err(4827):     at com.myapp.servercommunication.RequestSender.doInBackground(RequestSender.java:42)
    07-10 10:24:13.406: W/System.err(4827):     at com.myapp.servercommunication.RequestSender.doInBackground(RequestSender.java:1)
    07-10 10:24:13.406: W/System.err(4827):     at android.os.AsyncTask$2.call(AsyncTask.java:287)
    07-10 10:24:13.406: W/System.err(4827):     at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:305)
    07-10 10:24:13.406: W/System.err(4827):     at java.util.concurrent.FutureTask.run(FutureTask.java:137)
    07-10 10:24:13.406: W/System.err(4827):     at android.os.AsyncTask$SerialExecutor$1.run(AsyncTask.java:230)
    07-10 10:24:13.406: W/System.err(4827):     at java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1076)
    07-10 10:24:13.406: W/System.err(4827):     at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:569)
    07-10 10:24:13.406: W/System.err(4827):     at java.lang.Thread.run(Thread.java:856)
    07-10 10:24:13.409: W/System.err(4827): Caused by: java.io.EOFException
    07-10 10:24:13.413: W/System.err(4827):     at libcore.io.Streams.readAsciiLine(Streams.java:203)
    07-10 10:24:13.417: W/System.err(4827):     at libcore.net.http.HttpEngine.readResponseHeaders(HttpEngine.java:560)
    07-10 10:24:13.417: W/System.err(4827):     at libcore.net.http.HttpEngine.readResponse(HttpEngine.java:813)
    07-10 10:24:13.417: W/System.err(4827):     at libcore.net.http.HttpURLConnectionImpl.getResponse(HttpURLConnectionImpl.java:274)
    07-10 10:24:13.417: W/System.err(4827):     at libcore.net.http.HttpURLConnectionImpl.getResponseCode(HttpURLConnectionImpl.java:486)
    07-10 10:24:13.417: W/System.err(4827):     at org.springframework.http.client.SimpleClientHttpResponse.getRawStatusCode(SimpleClientHttpResponse.java:54)
    07-10 10:24:13.417: W/System.err(4827):     at org.springframework.http.client.SimpleClientHttpResponse.getStatusCode(SimpleClientHttpResponse.java:80)
    07-10 10:24:13.421: W/System.err(4827):     at org.springframework.web.client.DefaultResponseErrorHandler.hasError(DefaultResponseErrorHandler.java:46)
    07-10 10:24:13.421: W/System.err(4827):     at org.springframework.web.client.RestTemplate.doExecute(RestTemplate.java:477)
    07-10 10:24:13.421: W/System.
    err(4827):  ... 11 more

即使用例并不重要,我想知道为什么会抛出这个错误,我的实现是否有问题,或者它是 RestTemplate 的正常行为?

4

3 回答 3

5

服务器端毫无疑问java.io.EOFException。如果您在服务器端使用 keep-alive - 像这样关闭它:

 HttpHeaders headers = new HttpHeaders();
 headers.set("Connection", "Close");

或这个:System.setProperty("http.keepAlive", "false");

另一种解决方案是更改 http 客户端。在 Spring for Android 中,a 默认的 HTTP 客户端由RestTemplate 设备上的 Android 版本决定。API 9 或更新的用途HttpURLConnection,旧的用途HTTPClient。要将客户端显式设置为旧客户端,请使用 yourRestTemplate.setRequestFactory(new HttpComponentsClientHttpRequestFactory());

请参考:http ://static.springsource.org/spring-android/docs/1.0.1.RELEASE/reference/htmlsingle/#d4e34

SpringRestTemplate不是一个理想的 REST 客户端,不幸的是,它在生产代码中存在很多问题。

于 2013-07-10T09:18:23.233 回答
0

最近我在从 REST 模板发送请求时遇到了同样的问题。我们尝试了很多在 HTTP 标头中添加“http.keepAlive”,但它有时会起作用,有时它不起作用。最后,在使用以下代码设置“Accept-Language”标头后,我们能够解决此问题:

HttpHeaders headers = new HttpHeaders();
headers.set("Accept-Language", "en-US,en;q=0.8");
于 2014-04-28T10:45:08.260 回答
0

如果您使用新的 spring-android v.2.0.0.M3它需要包含 maven

        <dependency>
            <groupId>org.apache.httpcomponents</groupId>
            <artifactId>httpclient-android</artifactId>
            <version>4.3.5.1</version>
        </dependency>

或毕业

dependencies {
    compile('org.apache.httpcomponents:httpclient-android:4.3.5.1')
}

如果 Spring for Android 检测到 HttpClient 4.3,那么它会自动将其配置为默认 ClientHttpRequestFactory

于 2015-09-17T11:14:52.620 回答