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