由http调用引起的服务线程数高的问题处理

 

服务启动后线程数持续升高,重启后也是迅速上升,查看dump日志发现如下很多这种错误信息

"Connection evictor" #10756 daemon prio=5 os_prio=0 tid=0x00002ba4c102d000 nid=0x32ca waiting on condition [0x00002ba58ecc0000]
   java.lang.Thread.State: TIMED_WAITING (sleeping)
    at java.lang.Thread.sleep(Native Method)
    at org.apache.http.impl.client.IdleConnectionEvictor$1.run(IdleConnectionEvictor.java:66)
    at java.lang.Thread.run(Thread.java:745)

   Locked ownable synchronizers:
    - None

经过分析是由于使用httpclient进行http调用导致,代码中使用了如下方式创建httpclient客户端

 HttpClientBuilder builder = HttpClientBuilder.create().evictIdleConnections(5L, TimeUnit.SECONDS);

这种使用方法,每new一个httpclient,就会多创建一个idleconnection线程,IdleConnectionEvictor是为每个httpclient创建一个超时回收器,这个回收器就是一个单独的线程,但是这个IdleConnectionEvictor不会自动回收,所以随着httpclient创建的数据增多,这个IdleConnectionEvictor也会越来越多,最终导致线程数持续上升

List<Closeable> closeablesCopy = closeables != null ? new ArrayList<Closeable>(closeables) : null;
  if (!this.connManagerShared) {
   if (closeablesCopy == null) {
    closeablesCopy = new ArrayList<Closeable>(1);
   }
   final HttpClientConnectionManager cm = connManagerCopy;
 
   if (evictExpiredConnections || evictIdleConnections) {
    final IdleConnectionEvictor connectionEvictor = new IdleConnectionEvictor(cm,
      maxIdleTime > 0 ? maxIdleTime : 10, maxIdleTimeUnit != null ? maxIdleTimeUnit : TimeUnit.SECONDS,
      maxIdleTime, maxIdleTimeUnit);
    closeablesCopy.add(new Closeable() {
 
     @Override
     public void close() throws IOException {
      connectionEvictor.shutdown();
      try {
       connectionEvictor.awaitTermination(1L, TimeUnit.SECONDS);
      } catch (final InterruptedException interrupted) {
       Thread.currentThread().interrupt();
      }
     }
 
    });
    connectionEvictor.start();
   }
   closeablesCopy.add(new Closeable() {
 
    @Override
    public void close() throws IOException {
     cm.shutdown();
    }
 
   });
  }

解决方案有两个:

1.将httpclient变为static,只创建一个idleconnection线程,由它来负责全部httpclient的回收工作

static {
    Registry<ConnectionSocketFactory> registry = RegistryBuilder.create().register("http", PlainConnectionSocketFactory.getSocketFactory()).register("https", SSLConnectionSocketFactory.getSocketFactory()).build();
    PoolingHttpClientConnectionManager gcm = new PoolingHttpClientConnectionManager(registry);
    gcm.setMaxTotal(400);
    gcm.setDefaultMaxPerRoute(200);
    RequestConfig requestConfig = RequestConfig.custom().setConnectTimeout(5000).setSocketTimeout(30000).setConnectionRequestTimeout(5000).build();
    HttpClientBuilder httpClientBuilder = HttpClients.custom();
    httpClient = httpClientBuilder.setConnectionManager(gcm).setDefaultRequestConfig(requestConfig).setDefaultSocketConfig(SocketConfig.custom().setSoTimeout(30000).build()).evictExpiredConnections().evictIdleConnections(30L, TimeUnit.SECONDS).build();
    defaultHeaders = new Header[]{new BasicHeader("User-Agent", "Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/63.0.3239.132 Safari/537.36"), new BasicHeader("Accept", "text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,image/apng,*/*;q=0.8"), new BasicHeader("Accept-Encoding", "gzip, deflate"), new BasicHeader("Accept-Language", "zh-CN,zh;q=0.9"), new BasicHeader("Connection", "keep-alive"), new BasicHeader("Upgrade-Insecure-Requests", "1")};
} 

2.每次都创建一个新的,但是当自定义的client类被回收时,手动将httpClient涉及的东西都关闭

 if (httpClient != null) {
        try {
            httpClient.close();
        } catch (IOException e) {
            logger.error("browserHttpClient close exception", e);
        }
    }

 

 

 

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值