javax.net.ssl.SSLException: hostname in certificate didn't match:

本文详细介绍了如何解决在使用Apache HTTP库时遇到的SSL证书验证错误问题,包括设置自定义验证器和修改SSL上下文的方法。通过提供具体代码示例,帮助开发者顺利解决证书不匹配导致的连接失败。

抛出的错误是这样的:

[plain]  view plain copy print ?
  1. javax.net.ssl.SSLException: hostname in certificate didn't match: <test.rigbee.cn> != <rigbee.cn> OR <rigbee.cn> OR <www.rigbee.cn>  
  2.     at org.apache.http.conn.ssl.AbstractVerifier.verify(AbstractVerifier.java:185)  
  3.     at org.apache.http.conn.ssl.BrowserCompatHostnameVerifier.verify(BrowserCompatHostnameVerifier.java:54)  
  4.     at org.apache.http.conn.ssl.AbstractVerifier.verify(AbstractVerifier.java:114)  
  5.     at org.apache.http.conn.ssl.AbstractVerifier.verify(AbstractVerifier.java:95)  
  6.     at org.apache.http.conn.ssl.SSLSocketFactory.createSocket(SSLSocketFactory.java:381)  
  7.     at org.apache.http.impl.conn.DefaultClientConnectionOperator.openConnection(DefaultClientConnectionOperator.java:165)  
  8.     at org.apache.http.impl.conn.AbstractPoolEntry.open(AbstractPoolEntry.java:164)  
  9.     at org.apache.http.impl.conn.AbstractPooledConnAdapter.open(AbstractPooledConnAdapter.java:119)  
  10.     at org.apache.http.impl.client.DefaultRequestDirector.execute(DefaultRequestDirector.java:360)  
  11.     at org.apache.http.impl.client.AbstractHttpClient.execute(AbstractHttpClient.java:555)  
  12.     at org.apache.http.impl.client.AbstractHttpClient.execute(AbstractHttpClient.java:487)  
  13.     at org.apache.http.impl.client.AbstractHttpClient.execute(AbstractHttpClient.java:465)  

尝试过设置HttpsURLConnection.setDefaultHostnameVerifier(),不成功,估计是org.apache.http有自己的一套SSL东西,修改org.apache.http的主机名验证解决问题。

增加则用的一行:

[java]  view plain copy print ?
  1. SSLSocketFactory.getSocketFactory().setHostnameVerifier(new AllowAllHostnameVerifier());  

同时import一下:

[java]  view plain copy print ?
  1. import org.apache.http.conn.ssl.AllowAllHostnameVerifier;  
  2. import org.apache.http.conn.ssl.SSLSocketFactory;  

如果你使用org.apache.http的话上面的就够了。


另外如果你使用原始的URLConnect的话请这样设置一下:

[java]  view plain copy print ?
  1. try {         
  2.     SSLContext sc = SSLContext.getInstance("TLS");  
  3.     sc.init(nullnew TrustManager[] { new X509TrustManager() {  
  4.         @Override  
  5.         public void checkClientTrusted(X509Certificate[] chain, String authType) throws CertificateException {  
  6.         }  
  7.   
  8.   
  9.         @Override  
  10.         public void checkServerTrusted(X509Certificate[] chain, String authType)  
  11.   
  12.   
  13.         throws CertificateException {  
  14.         }  
  15.   
  16.   
  17.         @Override  
  18.         public X509Certificate[] getAcceptedIssuers() {  
  19.             return null;  
  20.         }  
  21.     } }, new SecureRandom());  
  22.     HttpsURLConnection.setDefaultSSLSocketFactory(sc.getSocketFactory());  
  23.     HttpsURLConnection.setDefaultHostnameVerifier(new HostnameVerifier() {  
  24.         @Override  
  25.         public boolean verify(String arg0, SSLSession arg1) {  
  26.             return true;  
  27.         }  
  28.     });  
  29. catch (Exception e) {  
  30.     e.printStackTrace();  
  31. }  


之后就可以使用了:

[java]  view plain copy print ?
  1. String https = "https://www.google.com.hk";  
  2. try {  
  3.     HttpsURLConnection conn = (HttpsURLConnection) new URL(https).openConnection();  
  4.     conn.setDoOutput(true);  
  5.     conn.setDoInput(true);  
  6.     conn.connect();  
  7.     BufferedReader br = new BufferedReader(new InputStreamReader(conn.getInputStream()));  
  8.     StringBuffer sb = new StringBuffer();  
  9.     String line;  
  10.     while ((line = br.readLine()) != null)  
  11.         sb.append(line);  
  12.     Log(sb.toString());  
  13.   
  14.   
  15. catch (Exception e) {  
  16.     e.printStackTrace();  
  17. }  
### 解决 `javax.net.ssl.SSLException: Connection reset` 异常 当遇到 `javax.net.ssl.SSLException: Connection reset` 异常时,这通常意味着客户端与服务器之间的SSL/TLS握手失败或连接被意外重置。以下是几种可能的原因及解决方案: #### 1. 验证HTTPS配置 确保所使用的URL确实是HTTPS URL,并且服务器支持TLS协议。如果尝试通过HTTP访问一个仅提供HTTPS的服务端口,则会引发此类错误。 ```java // 确认URL前缀为 "https" String url = "https://example.com/api"; ``` #### 2. 更新JVM的安全策略文件 有时默认安装的Java环境缺少必要的安全证书更新,可以通过下载最新的Java Cryptography Extension (JCE) Unlimited Strength Jurisdiction Policy Files来增强安全性设置[^1]。 #### 3. 调整信任库和密钥库路径 检查应用程序是否指定了正确的truststore位置以及密码。对于某些特定场景下运行的应用程序来说,可能还需要指定keystore的位置及其对应的密码。 ```properties -Djavax.net.ssl.trustStore=/path/to/truststore.jks \ -Djavax.net.ssl.trustStorePassword=changeit \ -Djavax.net.ssl.keyStore=/path/to/keystore.jks \ -Djavax.net.ssl.keyStorePassword=changeit ``` #### 4. 修改网络参数 调整TCP/IP栈的行为可以减少这种类型的异常发生频率。例如,在发起请求之前关闭Nagle算法(`TcpNoDelay`)或将套接字超时时间设得更短一些。 ```java builder.putHeaderParamsMap("Connection", "keep-alive"); HttpURLConnection conn = ...; conn.setUseCaches(false); conn.setRequestProperty("Keep-Alive", "timeout=15, max=100"); // 设置保持活动的时间和最大请求数量 ``` #### 5. 使用现代加密套件 确认双方都启用了兼容性强且较为安全的加密套件版本。旧版浏览器或应用可能会因为不支持新的TLS标准而无法建立稳定连接。 ```java System.setProperty("https.protocols", "TLSv1.2,TLSv1.3"); ``` 以上措施有助于缓解由不同原因引起的`javax.net.ssl.SSLException: Connection reset`问题。值得注意的是,具体实施哪项方案取决于实际环境中存在的根本原因分析结果。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值