HTTP客户端4.5不重复使用相同的连接使用java中的池连接管理器
问题描述:
我下面的代码是使用apache http客户端4.5版本的池连接管理器。如果我发出50个请求,我看到在netstat中有50个不同的tcp端口正在使用,但是在任何时候最多使用5个连接。我曾在线鲨中看到过滤器tcp.flags.syn==1 && tcp.flags.ack==0
它在过滤器中创建了50个数据包,因此它使用不同的连接而不是使用相同的连接,所以为什么不能这样做?HTTP客户端4.5不重复使用相同的连接使用java中的池连接管理器
我的代码:
import javax.net.ssl.SSLContext;
import org.apache.http.HttpEntity;
import org.apache.http.HttpHost;
import org.apache.http.client.ClientProtocolException;
import org.apache.http.client.methods.CloseableHttpResponse;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.config.Registry;
import org.apache.http.config.RegistryBuilder;
import org.apache.http.conn.routing.HttpRoute;
import org.apache.http.conn.socket.ConnectionSocketFactory;
import org.apache.http.conn.socket.PlainConnectionSocketFactory;
import org.apache.http.conn.ssl.NoopHostnameVerifier;
import org.apache.http.conn.ssl.SSLConnectionSocketFactory;
import org.apache.http.entity.StringEntity;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.HttpClients;
import org.apache.http.impl.conn.PoolingHttpClientConnectionManager;
import org.apache.http.util.EntityUtils;
import org.glassfish.jersey.SslConfigurator;
import org.json.JSONException;
import org.json.JSONObject;
public class App3Conn {
private static CloseableHttpClient client;
static String target="https://example.com";
static PoolingHttpClientConnectionManager cm ;
static{
SslConfigurator sslConfig = SslConfigurator.newInstance()
.securityProtocol("TLS")
.keyStoreFile("/Users/file")
.keyStorePassword("passw")
.keyStoreType("JKS")
.trustStoreFile("/Users/file");
SSLContext sslCtx = sslConfig.createSSLContext();
SSLConnectionSocketFactory sslsf = new SSLConnectionSocketFactory(sslCtx,NoopHostnameVerifier.INSTANCE);
Registry<ConnectionSocketFactory> r = RegistryBuilder.<ConnectionSocketFactory>create()
.register("http", PlainConnectionSocketFactory.getSocketFactory())
.register("https", sslsf)
.build();
cm = new PoolingHttpClientConnectionManager(r);
cm.setMaxTotal(15);
cm.setDefaultMaxPerRoute(5);
client = HttpClients.custom().setSSLSocketFactory(sslsf).setConnectionManager(cm).build();
}
public static void main(String a[]) throws JSONException, ClientProtocolException, IOException
{
JSONObject jsonMessage = new JSONObject();
JSONObject jsonResponse;
jsonMessage.put("keyID", "keyID");
StringEntity se = new StringEntity(jsonMessage.toString());
CloseableHttpResponse response2;
HttpPost httpPost = new HttpPost(target);
httpPost.setEntity(se);
httpPost.setHeader("Accept", "application/json");
httpPost.setHeader("Content-type", "application/json");
httpPost.setHeader("Connection", "keep-alive");
int i;
for(i=0;i<50;i++)
{
response2 = client.execute(httpPost);
HttpEntity entity2 = response2.getEntity();
String result = EntityUtils.toString(entity2);
EntityUtils.consume(entity2);
jsonResponse = new JSONObject(result);
String text = jsonResponse.getString("result");
response2.close();
}
}
}
答
我以前见过这一个。这取决于Apache连接池的工作方式。一个不透明的state
成员与池条目相关联,并且在请求(租用)池条目时必须匹配请求的状态。
当连接租赁尝试进行时,请求的state
是null
。但是,当连接放回池中时,state
将设置为SSL对等方的X500Principal
对象。这发生在DefaultUserTokenHandler
类中。幸运的是,当我们创建HttpClient
类时,这是可以覆盖的。这里有一个例子:
HttpClient apacheHttpClient = HttpClients.custom()
.setConnectionReuseStrategy(new DefaultConnectionReuseStrategy())
.setConnectionManager(new PoolingHttpClientConnectionManager(r))
.setUserTokenHandler(new UserTokenHandler() {
@Override
public Object getUserToken(HttpContext context) {
return null;
}
})
.build();
做检查,你没有失去你从DefaultUserTokenHandler
依赖和不使用此方法什么,如果你创建了HttpClient
可以通过相同的HTTP路径连接到多个SSL同行。在我的情况下,这个客户端用于连接到一个SSL服务器。
可能与您的TLS配置有关。我怀疑它认为它每次都需要一个新的连接,并没有意识到它可以重用连接。也许尝试如果删除它可以修复一些不同的服务器的症状。如果是这样,SSLConnectionSocketFactory可能是问题。我实际上并没有使用它,所以无法进一步帮助你。 –
检查哪一方正在关闭连接以及HTTP标头说什么。 Keep-Alive只是客户的建议,而不是服务器的必需品。请求完成后,服务器仍然可以关闭连接。 –
服务器有24小时保持活着财产 – Labeo