0


HttpUtils工具类(二)Apache HttpClient 5 使用详细教程

一、Apache HttpClient 5介绍

Apache HttpClient 5 是一个功能齐全且高度可定制的 HTTP 客户端库, 其专门用于发送 HTTP 请求、处理 HTTP 响应并支持各种 HTTP 协议特性。特别适合处理复杂的 HTTP 请求需求,如多协议支持、认证、连接池、代理等。它适合中大型项目或需要高级 HTTP 特性的应用开发。

(1)核心特性

  1. 功能强大:- 同步与异步支持:支持同步和异步的 HTTP 请求处理,异步请求在处理大量并发请求时能够显著提高效率。- 连接池管理:内置的连接池机制能提高并发处理性能,减少资源消耗。- 多种认证机制:支持多种认证方式,包括 Basic、Digest、NTLM、Kerberos 等,能够处理多种安全场景。- 支持Cookie管理:内置 Cookie 管理功能,能够自动处理服务端返回的 Cookie 并在后续请求中使用,模拟浏览器行为。
  2. 协议支持广泛:- HTTP/1.1 和 HTTP/2 支持:支持现代 HTTP 协议,包括 HTTP/2 的多路复用和流优先级等特性,提升了网络请求效率;特别是 HTTP/2 多路复用的优势。- SSL/TLS 支持:支持 HTTPS,提供自定义 SSL/TLS 配置,确保通信的安全性。HttpClient 5 可以方便地处理 HTTPS 请求,支持定制化的 SSL/TLS 配置。
  3. 灵活性和可扩展性:- 易于扩展和定制:允许开发者根据需要进行灵活的定制,HttpClient 5 的设计高度模块化,用户可以根据需要对连接管理、重定向策略、请求重试策略、代理设置等进行灵活定制。- 代理支持:可以轻松配置 HTTP 或 SOCKS 代理,用于跨网络访问和隐私保护。
  4. 健壮的错误处理机制:- 自动重试和重定向处理:内置的重试机制和自动重定向处理,可以减少由于网络问题导致的失败请求。开发者可以定制是否启用自动重定向。

(2)Apache HttpClient 5 的新特性

与之前的 4.x 版本相比,HttpClient 5 进行了大量的改进和优化,特别是在性能和安全性方面:

  • HTTP/2 支持:提供了完整的 HTTP/2 支持,包括多路复用、流优先级等特性。
  • 更好的异步支持:在处理并发请求时,通过异步模型极大提升了响应速度和吞吐量。
  • 灵活的响应处理:通过改进的响应处理 API,可以更方便地处理大型响应体,避免内存溢出问题。

(3)在 Java 项目的主要使用场景及缺点

使用场景:

  1. RESTful API 客户端:与第三方 API 交互,发送各种 HTTP 请求。
  2. Web 爬虫:抓取网页内容,处理重定向、Cookie 等。
  3. 分布式系统通信:用于微服务间的 HTTP 通信。
  4. 测试与自动化:模拟 HTTP 请求,进行集成和自动化测试。
  5. 代理与网关:处理请求代理和认证机制。
  6. 文件上传下载:实现大文件的传输。
  7. 认证交互:处理 OAuth2、JWT 等认证协议。
  8. 安全通信:处理 HTTPS 请求,确保数据安全。

缺点:

  1. 学习曲线陡峭:高级特性(如自定义连接池、SSL 配置、异步请求等)较为复杂,新手需要时间学习和掌握。
  2. 体积较大:相比轻量级客户端,如 OkHttp,HttpClient 依赖库较多,可能不适合小型项目。
  3. 性能消耗高:默认配置下的内存和 CPU 占用较高,需调整才能达到最佳性能。
  4. 配置繁琐:高级定制(如连接管理、认证、代理)需要较多配置,增加开发复杂度。
  5. 异步编程复杂:异步请求的回调、错误处理等逻辑复杂,增加代码难度。

二、在实际项目中的应用

(1)引入maven配置

首先,你需要将 HttpClient 5 的依赖加入到项目中。

  1. pom.xml

文件如下:

  1. <dependency>
  2. <groupId>org.apache.httpcomponents.client5</groupId>
  3. <artifactId>httpclient5</artifactId>
  4. <version>5.1</version>
  5. </dependency>

(2)自定义HttpUtils工具类---实现连接管理、重试策略等

  1. import lombok.extern.slf4j.Slf4j;
  2. import org.apache.hc.client5.http.DnsResolver;
  3. import org.apache.hc.client5.http.classic.methods.HttpPost;
  4. import org.apache.hc.client5.http.classic.methods.HttpUriRequestBase;
  5. import org.apache.hc.client5.http.config.RequestConfig;
  6. import org.apache.hc.client5.http.impl.classic.CloseableHttpClient;
  7. import org.apache.hc.client5.http.impl.classic.CloseableHttpResponse;
  8. import org.apache.hc.client5.http.impl.classic.HttpClients;
  9. import org.apache.hc.client5.http.impl.io.PoolingHttpClientConnectionManager;
  10. import org.apache.hc.client5.http.impl.io.PoolingHttpClientConnectionManagerBuilder;
  11. import org.apache.hc.core5.http.HttpEntity;
  12. import org.apache.hc.core5.http.io.SocketConfig;
  13. import org.apache.hc.core5.http.message.BasicHeader;
  14. import org.apache.hc.core5.util.Timeout;
  15. import org.springframework.stereotype.Component;
  16. import java.io.Closeable;
  17. import java.io.IOException;
  18. import java.net.InetAddress;
  19. import java.net.URI;
  20. import java.net.UnknownHostException;
  21. import java.util.Map;
  22. /**
  23. * @author 响叮当
  24. * @since 2024/8/15 14:10
  25. **/
  26. @Slf4j
  27. @Component
  28. public class ApacheHttpClientUtil {
  29. private CloseableHttpClient httpClient;
  30. @PostConstruct
  31. public void init() {
  32. SocketConfig socketConfig = SocketConfig.custom()
  33. .setSoTimeout(Timeout.ofMilliseconds(1000))
  34. .build();
  35. PoolingHttpClientConnectionManager connectionManager = PoolingHttpClientConnectionManagerBuilder
  36. .create()
  37. .setDefaultSocketConfig(socketConfig)
  38. .setMaxConnTotal(1000)
  39. .setMaxConnPerRoute(50)
  40. .build();
  41. RequestConfig requestConfig = RequestConfig.custom()
  42. .setConnectionRequestTimeout(Timeout.ofMilliseconds(8000))
  43. .setResponseTimeout(Timeout.ofMilliseconds(8000))
  44. .setConnectTimeout(Timeout.ofMilliseconds(8000))
  45. .build();
  46. httpClient = HttpClients
  47. .custom()
  48. .disableContentCompression()
  49. .setConnectionManager(connectionManager)
  50. .setDefaultRequestConfig(requestConfig)
  51. .build();
  52. }
  53. public CloseableHttpResponse getOrHead(String url, String method, Map<String, String> headers) throws IOException {
  54. HttpUriRequestBase request = new HttpUriRequestBase(method, URI.create(url));
  55. BasicHeader[] head = mapToHeaders(headers);
  56. request.setHeaders(head);
  57. return httpClient.execute(request);
  58. }
  59. public CloseableHttpResponse post(String url, Map<String, String> headers, HttpEntity httpEntity) throws IOException {
  60. HttpPost request = new HttpPost(url);
  61. BasicHeader[] head = mapToHeaders(headers);
  62. request.setHeaders(head);
  63. request.setEntity(httpEntity);
  64. return httpClient.execute(request);
  65. }
  66. public static BasicHeader[] mapToHeaders(Map<String, String> map) {
  67. BasicHeader[] headers = new BasicHeader[map.size()];
  68. int i = 0;
  69. for (Map.Entry<String, String> entry : map.entrySet()) {
  70. headers[i++] = new BasicHeader(entry.getKey(), entry.getValue());
  71. }
  72. return headers;
  73. }
  74. public static void closeQuietly(Closeable is) {
  75. if (is != null) {
  76. try {
  77. is.close();
  78. } catch (Exception ex) {
  79. log.error("Resources encounter an exception when closing,ex:{}", ex.getMessage());
  80. }
  81. }
  82. }
  83. }

功能概述

  1. 初始化 HTTP 客户端init() 方法初始化 CloseableHttpClient,配置连接池、请求超时和套接字超时。 - 连接池配置:通过 PoolingHttpClientConnectionManagerBuilder 设置最大连接数(1000)和每路由的最大连接数(50),用于高并发场景。- 请求配置:包括连接超时、请求超时和响应超时,确保在合理的时间内处理请求。
  2. HTTP 请求处理: - GET/HEAD 请求getOrHead() 方法可发送 GET 或 HEAD 请求,并接受自定义请求头。- POST 请求post() 方法可发送 POST 请求,支持自定义请求头和实体。
  3. 资源管理closeQuietly() 方法用于关闭资源,避免抛出异常。

(3)代码中的具体实现

1、Get请求

  1. @GetMapping("/get")
  2. public void testGet() {
  3. String url = "http://xxx.com.cn";
  4. try {
  5. // 1、构建入参、添加 headers
  6. Map<String, String> headers = new HashMap<>();
  7. // 2、发起http请求
  8. CloseableHttpResponse httpResponse = apacheHttpClientUtil.getOrHead(url, "GET", headers);
  9. // 3、返回结果,异常处理
  10. if (httpResponse.getCode() == 200) {
  11. String resStr = EntityUtils.toString(httpResponse.getEntity());
  12. log.info("request_success, response:{}, httpResponse_Code:{}, reasonPhrase:{}", resStr, httpResponse.getCode(), httpResponse.getReasonPhrase());
  13. } else {
  14. log.error("request_fail, httpResponse_Code:{}, reasonPhrase:{}", httpResponse.getCode(), httpResponse.getReasonPhrase());
  15. }
  16. } catch (Exception e) {
  17. log.error("request_udmp_fail, ex:{}", e);
  18. }
  19. }

2、POST请求-常规

  1. @PostMapping("/post")
  2. public void testPost(@RequestBody PostReq req) {
  3. String url = "http://xxx.com.cn";
  4. try {
  5. // 1、构建入参、添加 headers
  6. StringEntity stringEntity = new StringEntity(JSONUtil.toJsonStr(req), ContentType.APPLICATION_JSON);
  7. Map<String, String> headers = new HashMap<>();
  8. // 2、发起http请求
  9. CloseableHttpResponse httpResponse = apacheHttpClientUtil.post(url, headers, stringEntity);
  10. // 3、返回结果,异常处理
  11. if (httpResponse.getCode() == 200) {
  12. String resStr = EntityUtils.toString(httpResponse.getEntity());
  13. log.info("request_success, response:{}, httpResponse_Code:{}, reasonPhrase:{}", resStr, httpResponse.getCode(), httpResponse.getReasonPhrase());
  14. } else {
  15. log.error("request_fail, httpResponse_Code:{}, reasonPhrase:{}", httpResponse.getCode(), httpResponse.getReasonPhrase());
  16. }
  17. } catch (Exception e) {
  18. log.error("request_udmp_fail, ex:{}", e);
  19. }
  20. }

3、POST请求-上传文件

  1. /**
  2. * 上传文件
  3. * @param multipartFile
  4. * @param token
  5. * @param key
  6. * @return
  7. */
  8. @PostMapping("/uploadFile")
  9. public UploadRes testPostFile(
  10. @RequestParam("file") MultipartFile multipartFile,
  11. @RequestParam("token") String token,
  12. @RequestParam("key") String key
  13. ) {
  14. UploadRes res = null;
  15. String url = "http://xxx.com.cn";
  16. try {
  17. // 1、构建File参数
  18. File file = new File(multipartFile.getOriginalFilename());
  19. try (FileOutputStream fos = new FileOutputStream(file)) {
  20. fos.write(multipartFile.getBytes());
  21. }
  22. MultipartEntityBuilder builder = MultipartEntityBuilder.create();
  23. builder.addBinaryBody("file", file, ContentType.MULTIPART_FORM_DATA, "ex.xlsx");
  24. // 2、构建其他参数
  25. builder.addTextBody("token", token, ContentType.TEXT_PLAIN);
  26. builder.addTextBody("key", key, ContentType.TEXT_PLAIN);
  27. HttpEntity multipartEntity = builder.build();
  28. // 3、需要加 header,在这里加
  29. Map<String, String> headers = new HashMap<>();
  30. // 4、发起http请求
  31. CloseableHttpResponse httpResponse = apacheHttpClientUtil.post(url, headers, multipartEntity);
  32. // 5、返回结果,异常处理
  33. if (httpResponse.getCode() == 200) {
  34. String resStr = EntityUtils.toString(httpResponse.getEntity());
  35. res = JSONUtil.toBean(resStr, UploadRes.class);
  36. log.info("request_success, response:{}, httpResponse_Code:{}, reasonPhrase:{}", resStr, httpResponse.getCode(), httpResponse.getReasonPhrase());
  37. } else {
  38. log.error("request_fail, httpResponse_Code:{}, reasonPhrase:{}", httpResponse.getCode(), httpResponse.getReasonPhrase());
  39. }
  40. } catch (Exception e) {
  41. log.error("request_udmp_fail, ex:{}", e);
  42. }
  43. return res;
  44. }

(4)高级用法

1、处理重定向

HttpClient 5 默认会处理 3XX 重定向,但你也可以自定义行为。

  1. CloseableHttpClient httpClient = HttpClients.custom()
  2. .disableRedirectHandling() // 禁用自动重定向
  3. .build();

2、SSL/TLS 支持

使用 HttpClient 5 可以轻松处理 HTTPS 请求,下面展示如何自定义 SSL 配置。

  1. import org.apache.hc.core5.ssl.SSLContextBuilder;
  2. import org.apache.hc.client5.http.impl.classic.HttpClients;
  3. import javax.net.ssl.SSLContext;
  4. SSLContext sslContext = SSLContextBuilder.create()
  5. .loadTrustMaterial((chain, authType) -> true) // 信任所有证书
  6. .build();
  7. CloseableHttpClient httpClient = HttpClients.custom()
  8. .setSSLContext(sslContext)
  9. .build();

3、处理异步请求

如果你需要发送异步 HTTP 请求,可以使用

  1. HttpAsyncClient

  1. import org.apache.hc.client5.http.impl.async.CloseableHttpAsyncClient;
  2. import org.apache.hc.client5.http.impl.async.HttpAsyncClients;
  3. import org.apache.hc.core5.concurrent.FutureCallback;
  4. import org.apache.hc.client5.http.classic.methods.HttpGet;
  5. CloseableHttpAsyncClient asyncClient = HttpAsyncClients.createDefault();
  6. asyncClient.start();
  7. HttpGet request = new HttpGet("https://jsonplaceholder.typicode.com/posts/1");
  8. asyncClient.execute(request, new FutureCallback<>() {
  9. @Override
  10. public void completed(CloseableHttpResponse response) {
  11. System.out.println("Response received: " + response.getCode());
  12. }
  13. @Override
  14. public void failed(Exception ex) {
  15. System.out.println("Request failed: " + ex.getMessage());
  16. }
  17. @Override
  18. public void cancelled() {
  19. System.out.println("Request cancelled");
  20. }
  21. });

4、处理 Cookie

  1. import org.apache.hc.client5.http.cookie.BasicCookieStore;
  2. import org.apache.hc.client5.http.impl.classic.HttpClients;
  3. import org.apache.hc.client5.http.cookie.CookieStore;
  4. CookieStore cookieStore = new BasicCookieStore();
  5. CloseableHttpClient httpClient = HttpClients.custom()
  6. .setDefaultCookieStore(cookieStore)
  7. .build();

5、HTTPDNS支持

  1. // 当域名为www.baidu.com结尾时候,转到local的机器上
  2. String Target_IP= "127.0.0.1";
  3. String[] domains = new String[]{"www.baidu.com"};
  4. DnsResolver dnsResolver = new DnsResolver() {
  5. @Override
  6. public InetAddress[] resolve(String host) throws UnknownHostException {
  7. if (host.endsWith(domains[0]))) {
  8. return new InetAddress[]{InetAddress.getByName(Target_IP)};
  9. }
  10. return new InetAddress[0];
  11. }
  12. @Override
  13. public String resolveCanonicalHostname(String s) {
  14. return null;
  15. }
  16. };
  17. // @PostConstruct
  18. public void init() {
  19. SocketConfig socketConfig = SocketConfig.custom()
  20. .setSoTimeout(Timeout.ofMilliseconds(1000))
  21. .build();
  22. PoolingHttpClientConnectionManager connectionManager = PoolingHttpClientConnectionManagerBuilder
  23. .create()
  24. .setDnsResolver(dnsResolver) // 这里是HttpDns配置
  25. .setDefaultSocketConfig(socketConfig)
  26. .setMaxConnTotal(1000)
  27. .setMaxConnPerRoute(50)
  28. .build();

本文转载自: https://blog.csdn.net/m0_52767002/article/details/141229191
版权归原作者 响叮当! 所有, 如有侵权,请联系我们删除。

“HttpUtils工具类(二)Apache HttpClient 5 使用详细教程”的评论:

还没有评论