我在当前项目中使用com.sun.net.httpserver.HttpsServer来处理客户端身份验证等。目前,它仅打印出客户端地址/端口,以便我可以检查一个TCP连接是否用于多个请求(keep-alive)或是否使用新的为每个请求建立连接(因此每次都进行新的SSL握手)。当我使用FireFox对服务器发出多个请求时,我可以看到keep-alive正在工作。因此服务器部分可以很好地处理GET和POST请求。

如果我使用HttpURLConnection向服务器发出请求(在这种情况下使用,则没有 SSL),那么keep-alive也可以工作:对于多个顺序启动的请求,仅建立一个连接。

但是,如果我使用HttpsURLConnection(使用完全相同的代码,但使用 SSL),则keep-alive不再起作用。因此,尽管我使用相同的SSLContext(和SSLSocketFactory),但对于每个请求都建立了新的连接:

// URL myUrl = ...
// SSLContext mySsl = ...
HttpsURLConnection conn = (HttpsURLConnection) myUrl.openConnection();
conn.setUseCaches(false);
conn.setSSLSocketFactory(mySsl.getSocketFactory());

conn.setRequestMethod("POST");
// send Data
// receive Data

由于许多请求将导致许多SSL握手,这是一个真正的性能问题,我如何强制HttpsURLConnection使用keep-alive

更新(2012-04-02):
我没有每次都调用mySsl.getSocketFactory(),而是尝试缓存SSLSocketFactory。但是什么都没有改变。问题仍然存在。

最佳答案

我遇到了这个完全相同的问题,经过一些深入的调试,终于有了解决方案。

Http(s)UrlConnection在默认情况下确实会处理Keep-Alive,但是套接字必须处于非常特定的条件才能重用。

这些是:

  • 输入流必须完全消耗。您必须在输入流上调用read,直到它返回-1并关闭它。
  • 基础套接字上的设置必须使用完全相同的对象。
  • 完成后,您应该在Http(s)URLConnection上调用断开连接(是的,这是违反直觉的)。

  • 在上面的代码中,问题是:
    conn.setSSLSocketFactory(mySsl.getSocketFactory());
    

    在初始化期间将getSocketFactory()的结果保存到静态变量中,然后将其传递给conn.setSSLSocketFactory应该可以重用该套接字。

    关于java - HttpsUrlConnection和保持 Activity 状态,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/9943351/

    10-11 17:12