clearReferencesThreads XXX but has failed to stop it.

原创
2016/09/29 10:44
阅读数 633

停掉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之内退出那么就会出现错误,否则就会正常退出。

展开阅读全文
打赏
0
0 收藏
分享
加载中
更多评论
打赏
0 评论
0 收藏
0
分享
返回顶部
顶部