clearReferencesThreads XXX but has failed to stop it.

停掉Tomcat出现如下异常

org.apache.catalina.loader.WebappClassLoaderBase.clearReferencesThreads The web application [ROOT] appears to have started a thread named [pool-3-thread-4] but has failed to stop it. This is very likely to create a memory leak. Stack trace of thread:

sun.misc.Unsafe.park(Native Method)

java.util.concurrent.locks.LockSupport.park(LockSupport.java:175)

java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject.await(AbstractQueuedSynchronizer.java:2039)

java.util.concurrent.LinkedBlockingQueue.take(LinkedBlockingQueue.java:442)

java.util.concurrent.ThreadPoolExecutor.getTask(ThreadPoolExecutor.java:1067)

java.util.concurrent.ThreadPoolExecutor.runWorker(ThreadPoolExecutor.java:1127)

java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:617)

 

原因是线程池的线程被意外中断,这种不安全的方式退出可能导致业务中断,如何正确退出呢?

一开始我开始使用的线程池如下:

private static ExecutorService service = Executors.newFixedThreadPool(8);

然后只要这个线程池被使用,那么停掉服务器就出现如上的错误,看错误也知道线程池的线程是没有被正确的方式退出。

 

接着我又尝试使用如下方式:

private static ExecutorService service = new ThreadPoolExecutor(8, 8,

60L, TimeUnit.SECONDS,

new LinkedBlockingQueue<>());

结果一样,停掉服务器就会报错,因为核心线程数是永远留在池子里,其他大于池子的线程会在60秒空闲后退出,所以某些线程也会被不安全的方式打断。

 

最后把核心线程数改为0,如下:

private static ExecutorService service = new ThreadPoolExecutor(0, 8,

60L, TimeUnit.SECONDS,

new LinkedBlockingQueue<>());

如果调用了这个线程池,同时在60s之内退出那么就会出现错误,否则就会正常退出。

转载于:https://my.oschina.net/heweipo/blog/753091

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值