处理https访问时URL.openConnection()返回com.sun.net.ssl.HttpsURLConnection,各位大侠请帮忙看看
测试环境:
JDK 1.4_2 操作系统:win2000 应用服务器:WebSphere 5.1.0
生产环境
JKD 1.6_0_14 操作系统:linux 应用服务器:Jboss
发送https代码如下:
- Java code
public String postHttps(String urlString, String data, Logger logger) throws Exception { SSLContext sc = null; TrustManager[] trustAllCerts = new TrustManager[] { new javax.net.ssl.X509TrustManager() { public java.security.cert.X509Certificate[] getAcceptedIssuers() { return null; } public void checkClientTrusted( java.security.cert.X509Certificate[] certs, String authType) { } public void checkServerTrusted( java.security.cert.X509Certificate[] certs, String authType) { } } }; try { sc = SSLContext.getInstance("SSL"); sc.init(null, trustAllCerts, new java.security.SecureRandom()); HttpsURLConnection.setDefaultSSLSocketFactory(sc.getSocketFactory()); } catch (Exception e) { } HostnameVerifier hv = new HostnameVerifier() { public boolean verify(String urlHostName, SSLSession session) { return urlHostName.equals(session.getPeerHost()); } }; HttpsURLConnection.setDefaultHostnameVerifier(hv); OutputStreamWriter os = null; BufferedReader reader = null; StringBuffer sb = new StringBuffer(); URL u = new URL(urlString); HttpsURLConnection conn = (HttpsURLConnection) u.openConnection(); // conn.setRequestProperty("Content-Type", "text/xml"); conn.setDoOutput(true); conn.setFollowRedirects(true); // conn.setReadTimeout(30000); os = new OutputStreamWriter(conn.getOutputStream()); os.write(data); os.flush(); os.close(); return null; }在测试环境是可以正常的发送到指定的https地址上,而在生产环境上访问时会抛出以下错误:
java.lang.ClassCastException:com.sun.net.ssl.internal.www.protocol.https.HttpsURLConnectionOLDImpl cannot be case to javax.net.ssl.HttpsURLConnection.
是在HttpsURLConnection conn = (HttpsURLConnection) u.openConnection(); 处抛出。
之后用一下页面测试了下:
- Java code
<%@ page language="java" contentType="text/html; charset=GBK" pageEncoding="GBK"%><%@ page import="java.net.*" %><!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"><html><head><meta http-equiv="Content-Type" content="text/html; charset=GBK"><title>测试Https</title></head><body><% out.print("默认:<br/>"); URL aURL = new URL("https://www.verisign.com"); URLConnection urlConn = aURL.openConnection(); if (urlConn instanceof com.sun.net.ssl.HttpsURLConnection) { out.print("*** openConnection returns an instanceof com.sun.net.ssl.HttpsURLConnection<br/>"); } if (urlConn instanceof javax.net.ssl.HttpsURLConnection) { out.print("*** openConnection returns an instanceof javax.net.ssl.HttpsURLConnection<br/>"); } if (urlConn instanceof HttpURLConnection) { out.print("*** openConnection returns an instnace of HttpURLConnection<br/>"); }%></body></html>测试环境输出:
*** openConnection returns an instanceof javax.net.ssl.HttpsURLConnection
*** openConnection returns an instnace of HttpURLConnection
生产环境输出:
*** openConnection returns an instanceof com.sun.net.ssl.HttpsURLConnection
*** openConnection returns an instnace of HttpURLConnection
这就是问题所在,URL.openConnection() 在测试环境返回javax.net.ssl.HttpsURLConnection,而在生产环境则返回:com.sun.net.ssl.HttpsURLConnection
如何能将生产环境URL.openConnection()的返回从com.sun.net.ssl.HttpsURLConnection 变成javax.net.ssl.HttpsURLConnection ?
望做过Https访问的大侠们不吝赐教!
在线等 或联系qq 13386778 万分感谢!
[解决办法]
javax.net.ssl.HttpsURLConnection
com.sun.net.ssl.HttpsURLConnection
com.sun.net.ssl.internal.www.protocol.https.HttpsURLConnectionOLDImpl
这些类貌似都是在jdk的jsse.jar中,而这个com.sun.net.ssl.HttpsURLConnection应该是deprecated的了,代码都一样吧,仅仅是环境不同,具体实现类的选择貌似视环境而定,感觉是环境高度相关的一个case,能不能先试试把测试环境的jsse.jar换到生成环境中...
[解决办法]
[解决办法]
原因2,3楼说的很清楚了;
你可以这样改下,再测试:
- Java code
URLConnection = null; if (urlConn instanceof com.sun.net.ssl.HttpsURLConnection) { out.print("*** openConnection returns an instanceof com.sun.net.ssl.HttpsURLConnection<br/>");urlConn = aURL.openConnection(); } if (urlConn instanceof javax.net.ssl.HttpsURLConnection) { out.print("*** openConnection returns an instanceof javax.net.ssl.HttpsURLConnection<br/>");urlConn = aURL.openConnection(); } if (urlConn instanceof HttpURLConnection) { out.print("*** openConnection returns an instnace of HttpURLConnection<br/>");urlConn = aURL.openConnection(); }
[解决办法]
是不是你的生产环境上没有导入证书啊?你需要把证书导入到你java程序所运行的那个JDK里或者JRE里才可以,每换一个新环境或者证书有变更都需要重新导入证书的。
具体的步骤,你得先拿到你要访问的网站证书,用IE访问,在页面上右键-->Properties,点击“Certificates”按钮,Details选项卡中,点击“Copy to file”。拿到证书后,将证书导入到你所用的那个JDK里。
导入证书的步骤:cmd 进入<%JAVA_HOME%>\lib\bin\目录,或者<%JRE_HOME%>\bin\目录(实际看你的服务器运行时用的是哪个jre环境),执行命令:keytool -importcert -alias 068169 -keystore ..\lib\security\cacerts -file C:\sertificent\10.145.68.169.cer -trustcacerts
keytool.exe是JDK提供的证书导入工具,具体参数含义可以-help自己查一下,值得一提的是keystore和file两个命令,需要指定路径,一个是证书库路径,一个是需要导入的证书路径,两个路径都不能有空格,否则会出错。要么将JRE装到一个没空格的路径下,要么使用相对路径,要么使用<%环境变量%>lib\security\cacerts指定路径,只要保证没空格就行。执行命令,需要提供密码,默认密码是:changeit,这个密码可以进行修改。
证书导入成功后,启动服务,再次访问下应该就OK了。