`
m635674608
  • 浏览: 5029059 次
  • 性别: Icon_minigender_1
  • 来自: 南京
社区版块
存档分类
最新评论

转:Java中用HttpsURLConnection访问Https链接的问题

    博客分类:
  • java
 
阅读更多
在web应用交互过程中,有很多场景需要保证通信数据的安全;在前面也有好多篇文章介绍了在Web Service调用过程中用WS-Security来保证接口交互过程的安全性,值得注意的是,该种方式基于的传输协议仍然是Http,采用这种方式可扩 展性和数据交互效率比较高;另外一种实现方式就是用Https,他是在协议层对Http的再次封装,加入了SSL/TLS,采用该协议进行通信的数据全部 都会被加密,由于目前Web开发编程中对此都有了一定程度的封装,所以采用Https对外提供服务,除了证书以外,对编程能力的要求并不高,相对于前者门 槛较低,但是由于对双方通信的所有数据都进行加密,而且交互过程中还有多次握手等,所以效率较低;以下就介绍下在Java中访问Https链接时会出现的 一些问题;

在Java中要访问Https链接时,会用到一个关键类HttpsURLConnection;参见如下实现代码:
Java代码 复制代码 收藏代码
  1. // 创建URL对象   
  2. URL myURL = new URL("https://www.sun.com");   
  3.   
  4. // 创建HttpsURLConnection对象,并设置其SSLSocketFactory对象   
  5. HttpsURLConnection httpsConn = (HttpsURLConnection) myURL   
  6.         .openConnection();   
  7.   
  8. // 取得该连接的输入流,以读取响应内容   
  9. InputStreamReader insr = new InputStreamReader(httpsConn   
  10.         .getInputStream());   
  11.   
  12. // 读取服务器的响应内容并显示   
  13. int respInt = insr.read();   
  14. while (respInt != -1) {   
  15.     System.out.print((char) respInt);   
  16.     respInt = insr.read();   
  17. }  
           // 创建URL对象
           URL myURL = new URL("https://www.sun.com");

           // 创建HttpsURLConnection对象,并设置其SSLSocketFactory对象
           HttpsURLConnection httpsConn = (HttpsURLConnection) myURL
                   .openConnection();

           // 取得该连接的输入流,以读取响应内容
           InputStreamReader insr = new InputStreamReader(httpsConn
                   .getInputStream());

           // 读取服务器的响应内容并显示
           int respInt = insr.read();
           while (respInt != -1) {
               System.out.print((char) respInt);
               respInt = insr.read();
           }


在取得connection的时候和正常浏览器访问一样,仍然会验证服务端的证书是否被信任(权威机构发行或者被权威机构签名);如果服务端证书不被信任,则默认的实现就会有问题,一般来说,用SunJSSE会抛如下异常信息:
javax.net.ssl.SSLHandshakeException: sun.security.validator.ValidatorException: PKIX path building failed: sun.security.provider.certpath.SunCertPathBuilderException: unable to find valid certification path to requested target

上面提到SunJSSE,JSSE(Java Secure Socket Extension)是实现Internet安全通信的一系列包的集合。它是一个SSL和TLS的纯Java实现,可以透明地提供数据加密、服务器认证、 信息完整性等功能,可以使我们像使用普通的套接字一样使用JSSE建立的安全套接字。JSSE是一个开放的标准,不只是Sun公司才能实现一个 SunJSSE,事实上其他公司有自己实现的JSSE,然后通过JCA就可以在JVM中使用。
关于JSSE的详细信息参考官网Reference:http://java.sun.com/j2se/1.5.0/docs/guide/security/jsse/JSSERefGuide.html;
以及Java Security Guide:http://java.sun.com/j2se/1.5.0/docs/guide/security/;

在深入了解JSSE之前,需要了解一个有关Java安全的概念:客户端的TrustStore文件。客户端的TrustStore文件中保存着被客 户端所信任的服务器的证书信息。客户端在进行SSL连接时,JSSE将根据这个文件中的证书决定是否信任服务器端的证书。在SunJSSE中,有一个信任 管理器类负责决定是否信任远端的证书,这个类有如下的处理规则:
1、若系统属性javax.net.sll.trustStore指定了TrustStore文件,那么信任管理器就去jre安装路径下的lib/security/目录中寻找并使用这个文件来检查证书。
2、若该系统属性没有指定TrustStore文件,它就会去jre安装路径下寻找默认的TrustStore文件,这个文件的相对路径为:lib/security/jssecacerts。
3、若jssecacerts不存在,但是cacerts存在(它随J2SDK一起发行,含有数量有限的可信任的基本证书),那么这个默认的TrustStore文件就是lib/security/cacerts。

那遇到这种情况,怎么处理呢?有以下两种方案:
1、按照以上信任管理器的规则,将服务端的公钥导入到jssecacerts,或者是在系统属性中设置要加载的trustStore文件的路径;证书导入可以用如下命令:keytool -import -file src_cer_file –keystore dest_cer_store;至于证书可以通过浏览器导出获得;
2、实现自己的证书信任管理器类,比如MyX509TrustManager,该类必须实现X509TrustManager接口中的三个method;然后在HttpsURLConnection中加载自定义的类,可以参见如下两个代码片段,其一为自定义证书信任管理器,其二为connect时的代码:
Java代码 复制代码 收藏代码
  1.    package test;   
  2.   
  3.    import java.io.FileInputStream;   
  4.    import java.security.KeyStore;   
  5.    import java.security.cert.CertificateException;   
  6.    import java.security.cert.X509Certificate;   
  7.   
  8.    import javax.net.ssl.TrustManager;   
  9.    import javax.net.ssl.TrustManagerFactory;   
  10.    import javax.net.ssl.X509TrustManager;   
  11.   
  12.    public class MyX509TrustManager implements X509TrustManager {   
  13.   
  14.        /*  
  15.         * The default X509TrustManager returned by SunX509. We'll delegate  
  16.         * decisions to it, and fall back to the logic in this class if the  
  17.         * default X509TrustManager doesn't trust it.  
  18.         */  
  19.        X509TrustManager sunJSSEX509TrustManager;   
  20.   
  21.        MyX509TrustManager() throws Exception {   
  22.            // create a "default" JSSE X509TrustManager.   
  23.   
  24.            KeyStore ks = KeyStore.getInstance("JKS");   
  25.            ks.load(new FileInputStream("trustedCerts"),   
  26.                "passphrase".toCharArray());   
  27.   
  28.            TrustManagerFactory tmf =   
  29.            TrustManagerFactory.getInstance("SunX509""SunJSSE");   
  30.   
  31.             tmf.init(ks);   
  32.   
  33.          TrustManager tms [] = tmf.getTrustManagers();   
  34.   
  35.          /*  
  36.           * Iterate over the returned trustmanagers, look  
  37.           * for an instance of X509TrustManager. If found,  
  38.           * use that as our "default" trust manager.  
  39.           */  
  40.          for (int i = 0; i < tms.length; i++) {   
  41.              if (tms[i] instanceof X509TrustManager) {   
  42.                  sunJSSEX509TrustManager = (X509TrustManager) tms[i];   
  43.                  return;   
  44.              }   
  45.          }   
  46.   
  47.          /*  
  48.           * Find some other way to initialize, or else we have to fail the  
  49.           * constructor.  
  50.           */  
  51.          throw new Exception("Couldn't initialize");   
  52.      }   
  53.   
  54.      /*  
  55.       * Delegate to the default trust manager.  
  56.       */  
  57.      public void checkClientTrusted(X509Certificate[] chain, String authType)   
  58.                  throws CertificateException {   
  59.          try {   
  60.              sunJSSEX509TrustManager.checkClientTrusted(chain, authType);   
  61.          } catch (CertificateException excep) {   
  62.              // do any special handling here, or rethrow exception.   
  63.          }   
  64.      }   
  65.   
  66.      /*  
  67.       * Delegate to the default trust manager.  
  68.       */  
  69.      public void checkServerTrusted(X509Certificate[] chain, String authType)   
  70.                  throws CertificateException {   
  71.          try {   
  72.              sunJSSEX509TrustManager.checkServerTrusted(chain, authType);   
  73.          } catch (CertificateException excep) {   
  74.              /*  
  75.               * Possibly pop up a dialog box asking whether to trust the  
  76.               * cert chain.  
  77.               */  
  78.          }   
  79.      }   
  80.   
  81.      /*  
  82.       * Merely pass this through.  
  83.       */  
  84.      public X509Certificate[] getAcceptedIssuers() {   
  85.          return sunJSSEX509TrustManager.getAcceptedIssuers();   
  86.      }   
  87. }   
  88.   
  89.           // 创建SSLContext对象,并使用我们指定的信任管理器初始化   
  90.          TrustManager[] tm = { new MyX509TrustManager() };   
  91.          SSLContext sslContext = SSLContext.getInstance("SSL""SunJSSE");   
  92.   
  93.          sslContext.init(null, tm, new java.security.SecureRandom());   
  94.   
  95.          // 从上述SSLContext对象中得到SSLSocketFactory对象   
  96.          SSLSocketFactory ssf = sslContext.getSocketFactory();   
  97.   
  98.          // 创建URL对象   
  99.          URL myURL = new URL("https://ebanks.gdb.com.cn/sperbank/perbankLogin.jsp");   
  100.   
  101.          // 创建HttpsURLConnection对象,并设置其SSLSocketFactory对象   
  102.          HttpsURLConnection httpsConn = (HttpsURLConnection) myURL.openConnection();   
  103.          httpsConn.setSSLSocketFactory(ssf);   
  104.   
  105.          // 取得该连接的输入流,以读取响应内容   
  106.          InputStreamReader insr = new InputStreamReader(httpsConn.getInputStream());   
  107.   
  108.          // 读取服务器的响应内容并显示   
  109.          int respInt = insr.read();   
  110.          while (respInt != -1) {   
  111.              System.out.print((char) respInt);   
  112.              respInt = insr.read();   
  113.          }  
   package test;

   import java.io.FileInputStream;
   import java.security.KeyStore;
   import java.security.cert.CertificateException;
   import java.security.cert.X509Certificate;

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

   public class MyX509TrustManager implements X509TrustManager {

       /*
        * The default X509TrustManager returned by SunX509. We'll delegate
        * decisions to it, and fall back to the logic in this class if the
        * default X509TrustManager doesn't trust it.
        */
       X509TrustManager sunJSSEX509TrustManager;

       MyX509TrustManager() throws Exception {
           // create a "default" JSSE X509TrustManager.

           KeyStore ks = KeyStore.getInstance("JKS");
           ks.load(new FileInputStream("trustedCerts"),
               "passphrase".toCharArray());

           TrustManagerFactory tmf =
           TrustManagerFactory.getInstance("SunX509", "SunJSSE");

            tmf.init(ks);

         TrustManager tms [] = tmf.getTrustManagers();

         /*
          * Iterate over the returned trustmanagers, look
          * for an instance of X509TrustManager. If found,
          * use that as our "default" trust manager.
          */
         for (int i = 0; i < tms.length; i++) {
             if (tms[i] instanceof X509TrustManager) {
                 sunJSSEX509TrustManager = (X509TrustManager) tms[i];
                 return;
             }
         }

         /*
          * Find some other way to initialize, or else we have to fail the
          * constructor.
          */
         throw new Exception("Couldn't initialize");
     }

     /*
      * Delegate to the default trust manager.
      */
     public void checkClientTrusted(X509Certificate[] chain, String authType)
                 throws CertificateException {
         try {
             sunJSSEX509TrustManager.checkClientTrusted(chain, authType);
         } catch (CertificateException excep) {
             // do any special handling here, or rethrow exception.
         }
     }

     /*
      * Delegate to the default trust manager.
      */
     public void checkServerTrusted(X509Certificate[] chain, String authType)
                 throws CertificateException {
         try {
             sunJSSEX509TrustManager.checkServerTrusted(chain, authType);
         } catch (CertificateException excep) {
             /*
              * Possibly pop up a dialog box asking whether to trust the
              * cert chain.
              */
         }
     }

     /*
      * Merely pass this through.
      */
     public X509Certificate[] getAcceptedIssuers() {
         return sunJSSEX509TrustManager.getAcceptedIssuers();
     }
}

          // 创建SSLContext对象,并使用我们指定的信任管理器初始化
         TrustManager[] tm = { new MyX509TrustManager() };
         SSLContext sslContext = SSLContext.getInstance("SSL", "SunJSSE");

         sslContext.init(null, tm, new java.security.SecureRandom());

         // 从上述SSLContext对象中得到SSLSocketFactory对象
         SSLSocketFactory ssf = sslContext.getSocketFactory();

         // 创建URL对象
         URL myURL = new URL("https://ebanks.gdb.com.cn/sperbank/perbankLogin.jsp");

         // 创建HttpsURLConnection对象,并设置其SSLSocketFactory对象
         HttpsURLConnection httpsConn = (HttpsURLConnection) myURL.openConnection();
         httpsConn.setSSLSocketFactory(ssf);

         // 取得该连接的输入流,以读取响应内容
         InputStreamReader insr = new InputStreamReader(httpsConn.getInputStream());

         // 读取服务器的响应内容并显示
         int respInt = insr.read();
         while (respInt != -1) {
             System.out.print((char) respInt);
             respInt = insr.read();
         }


对于以上两种实现方式,各有各的优点,第一种方式不会破坏JSSE的安全性,但是要手工导入证书,如果服务器很多,那每台服务器的JRE都必须做相同的操作;第二种方式灵活性更高,但是要小心实现,否则可能会留下安全隐患;

转自:http://hi.baidu.com/wzbg/blog/item/258a59cb81da5c13be09e68e.html

分享到:
评论

相关推荐

    java 用HttpsURLConnection进行传递中文时错误总结

    在Java编程中,当涉及到网络通信,特别是使用HTTPS协议与远程服务器进行数据交换时,可能会遇到各种问题,尤其是在处理包含中文字符的数据时。本篇文章将深入探讨如何使用`HttpsURLConnection`来正确地发送和接收...

    Android使用HttpClient和HttpsUrlConnection两种方式访问https网站

    本文将详细介绍如何使用`HttpClient`和`HttpsURLConnection`两种方式来访问HTTPS网站,包括验证证书和不验证证书的实现方法。 ### 1. Android中的HttpClient `HttpClient`是Apache提供的一种HTTP客户端库,它支持...

    java访问https网址下载文件

    在整个过程中,需要捕获可能出现的异常,例如网络连接问题、证书问题等,确保程序的健壮性。 以上就是使用Java访问HTTPS网址并下载文件的基本步骤,包括处理可能需要的证书下载。需要注意的是,实际生产环境中,...

    java实现读取证书访问https接口

    在Java编程中,访问HTTPS(安全超文本传输协议)接口通常涉及到处理数字证书,这是为了确保数据传输的...这个过程确保了与服务器的安全连接,并能正确处理服务器的数字证书,从而在Java应用程序中安全地访问HTTPS接口。

    Java HttpsURLConnection的使用

    在Java编程中,`HttpsURLConnection`是用于处理HTTPS(安全超文本传输协议)连接的类,它是`HttpURLConnection`的子类。这篇文章将深入探讨如何使用`HttpsURLConnection`进行HTTPS通信,以及它在源码和工具开发中的...

    Https请求工具类,Java实现http请求,https请求

    本篇文章将详细介绍如何使用Java语言实现Http和Https请求的工具类,包括如何建立Https连接、如何实现Post请求、如何处理SSL验证等内容。 在Java中,使用HttpURLConnection类可以实现Http和Https请求,但是对于...

    Java Https工具类,Java Https Post请求

    - 然后,创建`HttpsURLConnection`对象,设置连接属性如超时、请求方法等。 - 接着,设置请求头,例如`Content-Type`为`application/x-www-form-urlencoded`表示发送表单数据。 - 将POST参数转换成字符串,写入...

    Java下载https文件并上传阿里云oss服务器

    Java下载HTTPS文件并上传到阿里云OSS服务器是常见的文件操作任务,特别是在开发云存储解决方案时。以下将详细讲解这个过程中的关键知识点。 首先,Java下载HTTPS文件涉及到网络请求和安全证书的处理。HTTPS...

    java client访问https server(客户端代码、服务器端配置)

    2. 客户端代码:在Java中,客户端通常使用`javax.net.ssl.HttpsURLConnection`类来创建和管理HTTPS连接。设置信任的根证书、服务器证书和私钥等关键步骤包括设置SSLSocketFactory和KeyManager。 3. 服务器端配置:...

    android 使用HttpsURLConnection方式的SSL双向认证

    此项目“android 使用HttpsURLConnection方式的SSL双向认证”着重讲解了如何在Android应用中通过HttpsURLConnection实现SSL的客户端和服务器端之间的双向身份验证,确保通信的隐私和完整性。 首先,我们需要理解SSL...

    Java发https请求证书问题

    ### Java 发送 HTTPS 请求时遇到证书问题解析 在进行网络通信的过程中,HTTPS 协议由于其安全性被广泛应用。本文将围绕“Java 发送 HTTPS 请求证书问题”这一主题展开讨论,通过对给定文件中的代码示例及操作步骤...

    java发送http/https请求(get/post)代码

    本文将详细讲解如何使用Java发送GET和POST请求,以及涉及的HTTPS安全连接。 首先,理解HTTP和HTTPS的区别至关重要。HTTP(超文本传输协议)是一种用于分发超媒体信息的应用层协议,而HTTPS(超文本传输安全协议)是...

    Java连接实现代码 (JAVA代码和VC代码 已更正)

    6. **HTTPS连接**:对于安全的HTTP(HTTPS),Java提供了`SSLSocket`和`SSLServerSocket`,以及使用`HttpsURLConnection`进行安全的HTTP通信。它们涉及到SSL/TLS协议,用于加密数据传输,确保数据的机密性和完整性。...

    java 调用https webservice实例及axis包

    这通常涉及到`javax.net.ssl.HttpsURLConnection`类的设置,例如设置`SSLSocketFactory`,处理证书问题。 4. **调用Web Service**:使用生成的客户端代理类,通过其方法调用Web Service。注意,由于是HTTPS,所以在...

    Java中的SSL及HTTPS协议实例源码

    在Java中,实现HTTPS连接主要涉及`javax.net.ssl`包下的类,如`SSLSocketFactory`和`HttpsURLConnection`。 让我们看看如何在Java中使用SSL和HTTPS协议的源码实例: ```java import javax.net.ssl....

    java调用HTTPS

    当一个客户端(如Java应用程序)连接到HTTPS服务器时,会经过握手过程,包括交换证书、协商加密算法、生成会话密钥等步骤。 要实现Java调用HTTPS,我们需要以下步骤: 1. **导入信任的证书**:服务器通常会提供...

    Java中如何通过https调用Webservice接口_测试环境.rar

    使用`HttpsURLConnection`或Apache HttpClient等库建立连接,设置请求方法(如GET或POST),然后发送请求。例如,使用`HttpsURLConnection`: ```java URL url = new URL("https://yourwebservicelink"); ...

    java访问https的类.rar

    在Java编程中,访问HTTPS(安全超文本传输协议)站点是常见的需求,特别是在开发网络应用时,确保数据传输的安全性至关重要。SSL(Secure Sockets Layer)和它的继任者TLS(Transport Layer Security)协议用于加密...

Global site tag (gtag.js) - Google Analytics