javax.net.ssl.SSLPeerUnverifiedException: No peer certificate

1.在HttpClient的方式,创建一个自定义的类,继承自org.apache.http.conn.ssl.SSLSocketFactory,而不使用org.apache.http.conn.ssl.SSLSocketFactory。

 

import java.io.IOException;
import java.net.Socket;
import java.net.UnknownHostException;
import java.security.KeyManagementException;
import java.security.KeyStore;
import java.security.KeyStoreException;
import java.security.NoSuchAlgorithmException;
import java.security.UnrecoverableKeyException;
import java.security.cert.CertificateException;
import java.security.cert.X509Certificate;

import javax.net.ssl.SSLContext;
import javax.net.ssl.TrustManager;
import javax.net.ssl.X509TrustManager;

import org.apache.http.conn.ssl.SSLSocketFactory;
public class MySSLSocketFactory extends SSLSocketFactory {
    SSLContext sslContext = SSLContext.getInstance("TLS");

    public MySSLSocketFactory(KeyStore truststore) throws NoSuchAlgorithmException, KeyManagementException, KeyStoreException, UnrecoverableKeyException {
        super(truststore);

        TrustManager tm = new X509TrustManager() {
            public void checkClientTrusted(X509Certificate[] chain, String authType) throws CertificateException {
            }

            public void checkServerTrusted(X509Certificate[] chain, String authType) throws CertificateException {
            }

            public X509Certificate[] getAcceptedIssuers() {
                return null;
            }
        };

        sslContext.init(null, new TrustManager[] { tm }, null);
    }

    @Override
    public Socket createSocket(Socket socket, String host, int port, boolean autoClose) throws IOException, UnknownHostException {
        return sslContext.getSocketFactory().createSocket(socket, host, port, autoClose);
    }

    @Override
    public Socket createSocket() throws IOException {
        return sslContext.getSocketFactory().createSocket();
    }
}

 

 2.创建HttpClient的实例时,使用这个类。

 

public HttpClient getNewHttpClient() {
    try {
        KeyStore trustStore = KeyStore.getInstance(KeyStore.getDefaultType());
        trustStore.load(null, null);

        SSLSocketFactory sf = new MySSLSocketFactory(trustStore);
        sf.setHostnameVerifier(SSLSocketFactory.ALLOW_ALL_HOSTNAME_VERIFIER);

        HttpParams params = new BasicHttpParams();
        HttpProtocolParams.setVersion(params, HttpVersion.HTTP_1_1);
        HttpProtocolParams.setContentCharset(params, HTTP.UTF_8);

        SchemeRegistry registry = new SchemeRegistry();
        registry.register(new Scheme("http", PlainSocketFactory.getSocketFactory(), 80));
        registry.register(new Scheme("https", sf, 443));

        ClientConnectionManager ccm = new ThreadSafeClientConnManager(params, registry);

        return new DefaultHttpClient(ccm, params);
    } catch (Exception e) {
        return new DefaultHttpClient();
    }
}
### 解决 `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`问题。值得注意的是,具体实施哪项方案取决于实际环境中存在的根本原因分析结果。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值