`
hkjdkl
  • 浏览: 31437 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

httpClient的3种超时

    博客分类:
  • java
阅读更多


你可以把HttpClient想象成一个浏览器,通过它的API我们可以很方便的发出GET,POST请求(当然它的功能远不止这些)。
  比如你只需以下几行代码就能发出一个简单的GET请求并打印响应结果:

  try {
          // 创建一个默认的HttpClient
          HttpClient httpclient = new DefaultHttpClient();
          // 创建一个GET请求
          HttpGet request = new HttpGet("www.google.com");
          // 发送GET请求,并将响应内容转换成字符串
          String response = httpclient.execute(request, new BasicResponseHandler());
          Log.v("response text", response);
      } catch (ClientProtocolException e) {
          e.printStackTrace();
      } catch (IOException e) {
          e.printStackTrace();
      }

  为什么要使用单例HttpClient?
  这只是一段演示代码,实际的项目中的请求与响应处理会复杂一些,并且还要考虑到代码的容错性,但是这并不是本篇的重点。注意代码的第三行:

  HttpClient httpclient = new DefaultHttpClient();

  在发出HTTP请求前,我们先创建了一个HttpClient对象。那么,在实际项目中,我们很可能在多处需要进行HTTP通信,这时候我们不需要为每个请求都创建一个新的HttpClient。因为之前已经提到,HttpClient就像一个小型的浏览器,对于整个应用,我们只需要一个HttpClient就够了。看到这里,一定有人心里想,这有什么难的,用单例啊!!就像这样:

  public class CustomerHttpClient {
      private static HttpClient customerHttpClient;
   
      private CustomerHttpClient() {
      }
   
      public static HttpClient getHttpClient() {
          if(null == customerHttpClient) {
              customerHttpClient = new DefaultHttpClient();
          }
          return customerHttpClient;
      }
  }

  多线程!试想,现在我们的应用程序使用同一个HttpClient来管理所有的Http请求,一旦出现并发请求,那么一定会出现多线程的问题。这就好像我们的浏览器只有一个标签页却有多个用户,A要上google,B要上baidu,这时浏览器就会忙不过来了。幸运的是,HttpClient提供了创建线程安全对象的API,帮助我们能很快地得到线程安全的“浏览器”。

  解决多线程问题

  public class CustomerHttpClient {
      private static final String CHARSET = HTTP.UTF_8;
      private static HttpClient customerHttpClient;

      private CustomerHttpClient() {
      }

      public static synchronized HttpClient getHttpClient() {
          if (null == customerHttpClient) {
              HttpParams params = new BasicHttpParams();
              // 设置一些基本参数
              HttpProtocolParams.setVersion(params, HttpVersion.HTTP_1_1);
              HttpProtocolParams.setContentCharset(params,
                      CHARSET);
              HttpProtocolParams.setUseExpectContinue(params, true);
              HttpProtocolParams
                      .setUserAgent(
                              params,
                              "Mozilla/5.0(Linux;U;Android 2.2.1;en-us;Nexus One Build.FRG83) "
                                      + "AppleWebKit/553.1(KHTML,like Gecko) Version/4.0 Mobile Safari/533.1");
              // 超时设置
              /* 从连接池中取连接的超时时间 */
              ConnManagerParams.setTimeout(params, 1000);
              /* 连接超时 */
              HttpConnectionParams.setConnectionTimeout(params, 2000);
              /* 请求超时 */
              HttpConnectionParams.setSoTimeout(params, 4000);
           
              // 设置我们的HttpClient支持HTTP和HTTPS两种模式
              SchemeRegistry schReg = new SchemeRegistry();
              schReg.register(new Scheme("http", PlainSocketFactory
                      .getSocketFactory(), 80));
              schReg.register(new Scheme("https", SSLSocketFactory
                      .getSocketFactory(), 443));

              // 使用线程安全的连接管理来创建HttpClient
              ClientConnectionManager conMgr = new ThreadSafeClientConnManager(
                      params, schReg);
              customerHttpClient = new DefaultHttpClient(conMgr, params);
          }
          return customerHttpClient;
      }
  }

  在上面的getHttpClient()方法中,我们为HttpClient配置了一些基本参数和超时设置,然后使用ThreadSafeClientConnManager来创建线程安全的HttpClient。上面的代码提到了3种超时设置,比较容易搞混,故在此特作辨析。

 


/* 从连接池中取连接的超时时间 */
  ConnManagerParams.setTimeout(params, 1000);
  /* 连接超时 */
  HttpConnectionParams.setConnectionTimeout(params, 2000);
  /* 请求超时 */
  HttpConnectionParams.setSoTimeout(params, 4000);

  第一行设置ConnectionPoolTimeout:这定义了从ConnectionManager管理的连接池中取出连接的超时时间,此处设置为1秒。
  第二行设置ConnectionTimeout:这定义了通过网络与服务器建立连接的超时时间。Httpclient包中通过一个异步线程去创建与服务器的socket连接,这就是该socket连接的超时时间,此处设置为2秒。
  第三行设置SocketTimeout:这定义了Socket读数据的超时时间,即从服务器获取响应数据需要等待的时间,此处设置为4秒。
  以上3种超时分别会抛出ConnectionPoolTimeoutException,ConnectionTimeoutException与SocketTimeoutException。

  封装简单的POST请求
  有了单例的HttpClient对象,我们就可以把一些常用的发出GET和POST请求的代码也封装起来,写进我们的工具类中了。目前我仅仅实现发出POST请求并返回响应字符串的方法以供大家参考。将以下代码加入我们的CustomerHttpClient类中:

  private static final String TAG = "CustomerHttpClient";

  public static String post(String url, NameValuePair... params) {
          try {
              // 编码参数
              List<NameValuePair> formparams = new ArrayList<NameValuePair>(); // 请求参数
              for (NameValuePair p : params) {
                  formparams.add(p);
              }
              UrlEncodedFormEntity entity = new UrlEncodedFormEntity(formparams,
                      CHARSET);
              // 创建POST请求
              HttpPost request = new HttpPost(url);
              request.setEntity(entity);
              // 发送请求
              HttpClient client = getHttpClient();
              HttpResponse response = client.execute(request);
              if(response.getStatusLine().getStatusCode() != HttpStatus.SC_OK) {
                  throw new RuntimeException("请求失败");
              }
              HttpEntity resEntity =  response.getEntity();
              return (resEntity == null) ? null : EntityUtils.toString(resEntity, CHARSET);
          } catch (UnsupportedEncodingException e) {
              Log.w(TAG, e.getMessage());
              return null;
          } catch (ClientProtocolException e) {
              Log.w(TAG, e.getMessage());
              return null;
          } catch (IOException e) {
              throw new RuntimeException("连接失败", e);
          }

      }

  使用我们的CustomerHttpClient工具类
  现在,在整个项目中我们都能很方便的使用该工具类来进行网络通信的业务代码编写了。下面的代码演示了如何使用username和password注册一个账户并得到新账户ID。

  final String url = "http://yourdomain/context/adduser";
      //准备数据
      NameValuePair param1 = new BasicNameValuePair("username", "张三");
      NameValuePair param2 = new BasicNameValuePair("password", "123456");
      int resultId = -1;
      try {
          // 使用工具类直接发出POST请求,服务器返回json数据,比如"{userid:12}"
          String response = CustomerHttpClient.post(url, param1, param2);
          JSONObject root = new JSONObject(response);
          resultId = Integer.parseInt(root.getString("userid"));
          Log.i(TAG, "新用户ID:" + resultId);
      } catch (RuntimeException e) {
          // 请求失败或者连接失败
          Log.w(TAG, e.getMessage());
          Toast.makeText(this, e.getMessage(), Toast.LENGTH_SHORT);
      } catch (Exception e) {
          // JSon解析出错
          Log.w(TAG, e.getMessage());
      }

 

分享到:
评论

相关推荐

    dotnet 6 精细控制 HttpClient 网络请求超时.rar

    3. **适当地配置超时**:根据实际需求设置超时时间,确保既不会因为过短的超时导致请求频繁失败,也不会因为过长的超时影响应用性能。 总之,理解并正确设置HttpClient的超时控制是优化网络请求性能的关键。通过...

    java httpclient设置超时时间和代理的方法

    Java HTTPClient 设置超时时间和代理的方法 Java HTTPClient 是一个功能强大且灵活的 HTTP 客户端库,提供了许多实用的功能来处理 HTTP 请求。其中,设置超时时间和代理是两个非常重要的配置项,本文将详细介绍如何...

    HttpClient4.5.1手册

    ### HttpClient4.5.1 手册知识点概览 #### 一、基础知识 **1.1 请求执行** - **HTTP请求:** 在客户端与服务器之间的交互过程中,客户端发送的每一条消息都称为一个请求。HTTP请求由三部分组成:请求行、请求头...

    dotnet 6 使用 HttpClient 的超时机制.rar

    本教程将详细讲解如何在.NET 6中利用HttpClient设置超时机制,确保你的请求处理在预设时间内完成,避免程序因长时间等待响应而卡死。 HttpClient类提供了发送HTTP请求的能力,同时允许开发者配置请求的多个方面,...

    httpclient.jar包下载

    3. **身份验证和安全**:HttpClient支持多种身份验证机制,包括基本认证、摘要认证、NTLM等,并且可以与SSL/TLS协议结合,确保数据传输的安全性。 4. **重试和恢复策略**:HttpClient内置了重试和恢复策略,当网络...

    httpClient需要的jar包

    2. **连接管理**:HttpClient提供了一种机制来管理HTTP连接,包括连接池(PoolingHttpClientConnectionManager),它可以重用已建立的TCP连接,提高性能并减少网络延迟。 3. **请求与响应**:HttpClient通过...

    httpclient3个jar包

    - **线程安全**:HttpClient 3.x版本不完全线程安全,如果在多线程环境中使用,需要正确管理和同步HttpClient实例。 - **连接管理**:HttpClient需要管理HTTP连接,包括连接池和超时设置,以优化性能和避免资源耗尽...

    httpclient httpclient.jar

    通常,我们创建一个HttpClient实例,并设置其配置参数,如连接超时、重试策略等。 2. **HttpHost**:表示目标HTTP服务器的信息,包括主机名、端口和协议(HTTP或HTTPS)。 3. **HttpGet/HttpPost**:这些是执行...

    HttpClientHelper 工具类

    单例模式是软件设计模式的一种,确保一个类只有一个实例,并提供一个全局访问点。HttpClientHelper 中的 SingleHelper 就是实现了单例模式的辅助类,为HttpClientHelper提供全局唯一的实例。这样做的好处是避免了...

    httpclient

    可以通过设置连接超时、读写超时、最大连接数、连接复用策略等来优化HttpClient的性能。 10. **异步操作**: HttpClient也支持异步操作,通过`HttpAsyncClient`类,可以在回调函数中处理响应,适合于高并发场景。...

    Httpclient依赖包

    3. **身份验证**:HttpClient支持多种身份验证机制,包括基本认证、摘要认证、NTLM、Kerberos等,能够处理跨域和多层代理的认证问题。 4. **Cookie管理**:HttpClient内置了Cookie管理器,可以处理服务器返回的...

    httpClienthttpClient

    10. **Timeouts**:可以设置连接超时、读取超时和写入超时,以防止程序因网络问题而阻塞。 在实际使用中,你通常会先创建一个HttpClient实例,然后创建HttpRequestBase对象,设置请求参数,再通过HttpClient发送...

    commons-httpclient-3.0.jar JAVA中使用HttpClient可以用到

    1. **创建HttpClient实例**:首先,需要创建一个HttpClient实例,可以设置连接超时、重试策略等参数。 2. **选择HttpMethod**:根据实际需求选择合适的HttpMethod,比如使用GetMethod或PostMethod。 3. **设置请求...

    httpclient4.2.1.zip

    除了基本的HTTP请求,HttpClient还支持更复杂的操作,如文件上传、下载、分块传输、自定义HTTP头、超时设置等。例如,使用HttpPost进行表单数据提交: ```java HttpPost httpPost = new HttpPost(...

    HttpClient重新封装的HttpClient类

    然而,这个基础版本的`HttpClient`存在一些问题,如资源管理不当可能导致泄露,以及缺少重试和超时策略等。因此,对`HttpClient`进行重新封装是非常常见的实践,以解决这些问题并增加额外的功能。 重新封装的`...

    httpclient-4.5jar包

    3. 执行请求:通过HttpClient对象的execute方法发送请求,获取HttpResponse。 4. 处理响应:解析HttpResponse,获取状态码、响应头和响应体。 四、连接管理和池化 HttpClient 4.5提供了...

    apache httpclient jar包

    3. **请求和响应头**: - 可以通过HttpUriRequest.setHeader()和HttpResponse.getAllHeaders()来设置和获取请求头和响应头。 4. **连接管理**: - **ConnectionManager**:负责管理TCP连接,包括建立、复用和关闭...

    Arduino HttpClient 库文件

    3. **设置请求方法和资源**:根据需求选择 GET 或 POST 方法。对于 GET,直接发送请求即可;对于 POST,需要先设置 POST 参数,然后调用 `client.post()`。 4. **发送请求**:调用 `client.sendRequest()` 或 `...

Global site tag (gtag.js) - Google Analytics