HttpClientUtils 使用 jdk 自带的 http5 替代 apache httpclient 工具

This commit is contained in:
thinkgem
2025-03-27 14:39:30 +08:00
parent c7cc1343c8
commit 6324fa15ae
4 changed files with 180 additions and 170 deletions

View File

@@ -87,13 +87,6 @@
<version>${snakeyaml.version}</version> <version>${snakeyaml.version}</version>
</dependency> </dependency>
<!-- Apache HTTP -->
<dependency>
<groupId>org.apache.httpcomponents</groupId>
<artifactId>httpclient</artifactId>
<version>${httpclient.version}</version>
</dependency>
<!-- Email --> <!-- Email -->
<dependency> <dependency>
<groupId>org.apache.commons</groupId> <groupId>org.apache.commons</groupId>

View File

@@ -4,218 +4,225 @@
*/ */
package com.jeesite.common.web.http; package com.jeesite.common.web.http;
import java.io.IOException; import com.jeesite.common.codec.EncodeUtils;
import java.io.UnsupportedEncodingException; import com.jeesite.common.lang.StringUtils;
import java.security.GeneralSecurityException; import org.slf4j.Logger;
import java.security.cert.CertificateException; import org.slf4j.LoggerFactory;
import javax.net.ssl.SSLContext;
import javax.net.ssl.SSLEngine;
import javax.net.ssl.TrustManager;
import javax.net.ssl.X509ExtendedTrustManager;
import java.net.Socket;
import java.net.URI;
import java.net.http.HttpClient;
import java.net.http.HttpRequest;
import java.net.http.HttpResponse;
import java.security.SecureRandom;
import java.security.cert.X509Certificate; import java.security.cert.X509Certificate;
import java.util.ArrayList; import java.time.Duration;
import java.util.List;
import java.util.Map; import java.util.Map;
import javax.net.ssl.HostnameVerifier;
import javax.net.ssl.SSLContext;
import javax.net.ssl.SSLSession;
import org.apache.http.HttpEntity;
import org.apache.http.NameValuePair;
import org.apache.http.client.entity.UrlEncodedFormEntity;
import org.apache.http.client.methods.CloseableHttpResponse;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.client.methods.HttpUriRequest;
import org.apache.http.conn.ssl.SSLConnectionSocketFactory;
import org.apache.http.conn.ssl.TrustStrategy;
import org.apache.http.entity.StringEntity;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.HttpClients;
import org.apache.http.message.BasicNameValuePair;
import org.apache.http.ssl.SSLContextBuilder;
import org.apache.http.util.EntityUtils;
import com.jeesite.common.codec.EncodeUtils;
/** /**
* HTTP客户端工具类支持HTTPS * HTTP 客户端工具类支持HTTPS
* @author ThinkGem * @author ThinkGem
* @version 2017-3-27 * @version 2025-03-26
*/ */
public class HttpClientUtils { public class HttpClientUtils {
private final static Logger logger = LoggerFactory.getLogger(HttpClientUtils.class);
private static final HttpClient client = createHttpClient(60);
/** /**
* http的get请求 * HTTP 的 GET 请求
* @param url
*/ */
public static String get(String url) { public static String get(String url) {
return get(url, EncodeUtils.UTF_8); return get(url, null, null);
}
/**
* http的get请求
* @param url
*/
public static String get(String url, String charset) {
HttpGet httpGet = new HttpGet(url);
return executeRequest(httpGet, charset);
}
/**
* http的get请求增加异步请求头参数
* @param url
*/
public static String ajaxGet(String url) {
return ajaxGet(url, EncodeUtils.UTF_8);
}
/**
* http的get请求增加异步请求头参数
* @param url
*/
public static String ajaxGet(String url, String charset) {
HttpGet httpGet = new HttpGet(url);
httpGet.setHeader("X-Requested-With", "XMLHttpRequest");
return executeRequest(httpGet, charset);
} }
/** /**
* http的post请求,传递map格式参数 * HTTP 的 GET 请求,传递 Map 格式参数
*/
public static String get(String url, Map<String, String> dataMap) {
return get(url, dataMap, EncodeUtils.UTF_8);
}
/**
* HTTP 的 GET 请求,传递 Map 格式参数,支持指定编码
*/
public static String get(String url, Map<String, String> dataMap, String charset) {
HttpRequest request = HttpRequest.newBuilder()
.uri(URI.create(buildUrl(url, dataMap, charset)))
.GET()
.build();
return executeRequest(request);
}
/**
* HTTP 的 GET 请求,增加 ajax 请求头
*/
public static String ajaxGet(String url) {
return ajaxGet(url, null, null);
}
/**
* HTTP 的 GET 请求,增加 ajax 请求头,传递 Map 格式参数
*/
public static String ajaxGet(String url, Map<String, String> dataMap) {
return ajaxGet(url, dataMap, EncodeUtils.UTF_8);
}
/**
* HTTP 的 GET 请求,增加 ajax 请求头,传递 Map 格式参数,支持指定编码
*/
public static String ajaxGet(String url, Map<String, String> dataMap, String charset) {
HttpRequest request = HttpRequest.newBuilder()
.uri(URI.create(buildUrl(url, dataMap, charset)))
.header("X-Requested-With", "XMLHttpRequest")
.GET()
.build();
return executeRequest(request);
}
/**
* 构建表单数据Map 转换 params支持指定编码
*/
private static String buildUrl(String url, Map<String, String> dataMap, String charset) {
if (dataMap == null) {
return url;
}
StringBuilder sb = new StringBuilder(url);
if (!url.contains("?")) {
sb.append("?");
} else if (!url.endsWith("&")) {
sb.append("&");
}
return sb + buildFormData(dataMap, charset);
}
/**
* HTTP 的 POST 请求,传递 Map 格式参数
*/ */
public static String post(String url, Map<String, String> dataMap) { public static String post(String url, Map<String, String> dataMap) {
return post(url, dataMap, EncodeUtils.UTF_8); return post(url, dataMap, EncodeUtils.UTF_8);
} }
/** /**
* http的post请求,传递map格式参数 * HTTP 的 POST 请求,传递 Map 格式参数,支持指定编码
*/ */
public static String post(String url, Map<String, String> dataMap, String charset) { public static String post(String url, Map<String, String> dataMap, String charset) {
HttpPost httpPost = new HttpPost(url); HttpRequest request = HttpRequest.newBuilder()
try { .uri(URI.create(url))
if (dataMap != null){ .header("Content-Type", "application/x-www-form-urlencoded")
List<NameValuePair> nvps = new ArrayList<NameValuePair>(); .POST(HttpRequest.BodyPublishers.ofString(buildFormData(dataMap, charset)))
for (Map.Entry<String, String> entry : dataMap.entrySet()) { .build();
nvps.add(new BasicNameValuePair(entry.getKey(), entry.getValue())); return executeRequest(request);
}
UrlEncodedFormEntity formEntity = new UrlEncodedFormEntity(nvps, charset);
formEntity.setContentEncoding(charset);
httpPost.setEntity(formEntity);
}
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}
return executeRequest(httpPost, charset);
} }
/** /**
* http的post请求增加异步请求头参数,传递map格式参数 * HTTP 的 POST 请求,增加 ajax 请求头,传递 Map 格式参数
*/ */
public static String ajaxPost(String url, Map<String, String> dataMap) { public static String ajaxPost(String url, Map<String, String> dataMap) {
return ajaxPost(url, dataMap, EncodeUtils.UTF_8); return ajaxPost(url, dataMap, EncodeUtils.UTF_8);
} }
/** /**
* http的post请求增加异步请求头参数,传递map格式参数 * HTTP 的 POST 请求,增加 ajax 请求头,传递 Map 格式参数,支持指定编码
*/ */
public static String ajaxPost(String url, Map<String, String> dataMap, String charset) { public static String ajaxPost(String url, Map<String, String> dataMap, String charset) {
HttpPost httpPost = new HttpPost(url); HttpRequest request = HttpRequest.newBuilder()
httpPost.setHeader("X-Requested-With", "XMLHttpRequest"); .uri(URI.create(url))
try { .header("X-Requested-With", "XMLHttpRequest")
if (dataMap != null){ .header("Content-Type", "application/x-www-form-urlencoded")
List<NameValuePair> nvps = new ArrayList<NameValuePair>(); .POST(HttpRequest.BodyPublishers.ofString(buildFormData(dataMap, charset)))
for (Map.Entry<String, String> entry : dataMap.entrySet()) { .build();
nvps.add(new BasicNameValuePair(entry.getKey(), entry.getValue())); return executeRequest(request);
}
UrlEncodedFormEntity formEntity = new UrlEncodedFormEntity(nvps, charset);
formEntity.setContentEncoding(charset);
httpPost.setEntity(formEntity);
}
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}
return executeRequest(httpPost, charset);
} }
/** /**
* http的post请求增加异步请求头参数传递json格式参数 * 构建表单数据Map 转换 params支持指定编码
*/
private static String buildFormData(Map<String, String> dataMap, String charset) {
return dataMap.entrySet().stream()
.map(entry -> entry.getKey() + "="
+ EncodeUtils.encodeUrl(entry.getValue(), charset))
.reduce((a, b) -> a + "&" + b)
.orElse(StringUtils.EMPTY);
}
/**
* HTTP 的 POST 请求,使用 json 请求头,传递 json 格式参数
*/ */
public static String ajaxPostJson(String url, String jsonString) { public static String ajaxPostJson(String url, String jsonString) {
return ajaxPostJson(url, jsonString, EncodeUtils.UTF_8); return ajaxPostJson(url, jsonString, EncodeUtils.UTF_8);
} }
/** /**
* http的post请求增加异步请求头参数传递json格式参数 * HTTP 的 POST 请求,使用 json 请求头,传递 json 格式参数,支持指定编码
*/ */
public static String ajaxPostJson(String url, String jsonString, String charset) { public static String ajaxPostJson(String url, String jsonString, String charset) {
HttpPost httpPost = new HttpPost(url); HttpRequest request = HttpRequest.newBuilder()
httpPost.setHeader("X-Requested-With", "XMLHttpRequest"); .uri(URI.create(url))
// try { .header("X-Requested-With", "XMLHttpRequest")
StringEntity stringEntity = new StringEntity(jsonString, charset);// 解决中文乱码问题 .header("Content-Type", "application/json; charset="
stringEntity.setContentEncoding(charset); + (StringUtils.isNotBlank(charset) ? charset : EncodeUtils.UTF_8))
stringEntity.setContentType("application/json"); .POST(HttpRequest.BodyPublishers.ofString(jsonString))
httpPost.setEntity(stringEntity); .build();
// } catch (UnsupportedEncodingException e) { return executeRequest(request);
// e.printStackTrace();
// }
return executeRequest(httpPost, charset);
} }
/** /**
* 执行一个http请求传递HttpGet或HttpPost参数 * 执行一个 http 请求,传递 HttpRequest 参数
*/ */
public static String executeRequest(HttpUriRequest httpRequest) { public static String executeRequest(HttpRequest request) {
return executeRequest(httpRequest, EncodeUtils.UTF_8); try {
return client.send(request, HttpResponse.BodyHandlers.ofString()).body();
} catch (Exception e) {
throw new RuntimeException(e);
}
} }
/** public static HttpClient createHttpClient(long seconds) {
* 执行一个http请求传递HttpGet或HttpPost参数 HttpClient client;
*/
public static String executeRequest(HttpUriRequest httpRequest, String charset) {
CloseableHttpClient httpclient;
if ("https".equals(httpRequest.getURI().getScheme())){
httpclient = createSSLInsecureClient();
}else{
httpclient = HttpClients.createDefault();
}
String result = "";
try { try {
try { SSLContext sslContext = SSLContext.getInstance("TLS");
CloseableHttpResponse response = httpclient.execute(httpRequest); sslContext.init(null, new TrustManager[]{new UnsafeX509ExtendedTrustManager()}, new SecureRandom());
HttpEntity entity = null; client = HttpClient.newBuilder()
try { .sslContext(sslContext)
entity = response.getEntity(); .connectTimeout(Duration.ofSeconds(seconds))
result = EntityUtils.toString(entity, charset); .build();
} finally { } catch (Exception e) {
EntityUtils.consume(entity); logger.info(e.getMessage(), e);
response.close(); client = HttpClient.newHttpClient();
}
} finally {
httpclient.close();
}
}catch(IOException ex){
ex.printStackTrace();
} }
return result; return client;
} }
/** private static final class UnsafeX509ExtendedTrustManager extends X509ExtendedTrustManager {
* 创建 SSL连接
*/ private static final X509Certificate[] EMPTY_CERTIFICATES = new X509Certificate[0];
public static CloseableHttpClient createSSLInsecureClient() {
try { @Override
SSLContext sslContext = new SSLContextBuilder().loadTrustMaterial(new TrustStrategy() { public void checkClientTrusted(X509Certificate[] certificates, String authType) {
@Override }
public boolean isTrusted(X509Certificate[] chain, String authType) throws CertificateException { @Override
return true; public void checkClientTrusted(X509Certificate[] certificates, String authType, Socket socket) {
} }
}).build(); @Override
SSLConnectionSocketFactory sslsf = new SSLConnectionSocketFactory(sslContext, new HostnameVerifier() { public void checkClientTrusted(X509Certificate[] certificates, String authType, SSLEngine sslEngine) {
@Override }
public boolean verify(String hostname, SSLSession session) { @Override
return true; public void checkServerTrusted(X509Certificate[] certificates, String authType) {
} }
}); @Override
return HttpClients.custom().setSSLSocketFactory(sslsf).build(); public void checkServerTrusted(X509Certificate[] certificates, String authType, Socket socket) {
} catch (GeneralSecurityException ex) { }
throw new RuntimeException(ex); @Override
public void checkServerTrusted(X509Certificate[] certificates, String authType, SSLEngine sslEngine) {
}
@Override
public X509Certificate[] getAcceptedIssuers() {
return EMPTY_CERTIFICATES;
} }
} }

View File

@@ -4,8 +4,12 @@
*/ */
package com.jeesite.test.web.http; package com.jeesite.test.web.http;
import com.jeesite.common.mapper.JsonMapper;
import com.jeesite.common.web.http.HttpClientUtils; import com.jeesite.common.web.http.HttpClientUtils;
import java.util.HashMap;
import java.util.Map;
/** /**
* HTTP客户端测试工具类支持HTTPS * HTTP客户端测试工具类支持HTTPS
* @author ThinkGem * @author ThinkGem
@@ -14,8 +18,15 @@ import com.jeesite.common.web.http.HttpClientUtils;
public class HttpClientUtilsTest { public class HttpClientUtilsTest {
public static void main(String[] args) { public static void main(String[] args) {
String content = HttpClientUtils.get("https://jeesite.com"); String url = "https://vue.jeesite.com/js/a/sys/corpAdmin/treeData";
System.out.println(content); Map<String, String> dataMap = new HashMap<>();
dataMap.put("isShowCode", "true");
dataMap.put("param1", "你好");
System.out.println(HttpClientUtils.get(url, dataMap));
System.out.println(HttpClientUtils.ajaxGet(url, dataMap));
System.out.println(HttpClientUtils.post(url, dataMap));
System.out.println(HttpClientUtils.ajaxPost(url, dataMap));
System.out.println(HttpClientUtils.ajaxPostJson(url, JsonMapper.toJson(dataMap)));
} }
} }

View File

@@ -31,7 +31,6 @@
<fastjson.version>2.0.51</fastjson.version> <fastjson.version>2.0.51</fastjson.version>
<snakeyaml.version>2.2</snakeyaml.version> <snakeyaml.version>2.2</snakeyaml.version>
<glassfish-jaxb.version>2.3.9</glassfish-jaxb.version> <glassfish-jaxb.version>2.3.9</glassfish-jaxb.version>
<httpclient.version>4.5.14</httpclient.version>
<UserAgentUtils.version>1.21</UserAgentUtils.version> <UserAgentUtils.version>1.21</UserAgentUtils.version>
<metadata-extractor.version>2.19.0</metadata-extractor.version> <metadata-extractor.version>2.19.0</metadata-extractor.version>