线程池关闭不合理,导致应用无法正常stop的情况

在上一篇博客中,我使用了线程池进行管理线程,达到线程复用的效果。

详情参考:线程池+CountDownLatch优化代码,提高程序执行效率

 

程序启动、运行皆无异常,线程池确实对程序中创建的线程进行管理,但是,在我关闭tomcat时,无法正常关闭,程序出现报错。报错信息如下:

05-Apr-2020 19:09:45.003 璀﹀憡 [localhost-startStop-2] org.apache.catalina.loader.WebappClassLoaderBase.clearReferencesThreads 
The web application [ROOT] appears to have started a thread named [pool-5-thread-2] 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.LinkedBlockingDeque.takeFirst(LinkedBlockingDeque.java:492)
 java.util.concurrent.LinkedBlockingDeque.take(LinkedBlockingDeque.java:680)
 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)
 java.lang.Thread.run(Thread.java:748

 注意看这行报错信息:

The web application [ROOT] appears to have started a thread named [pool-5-thread-2] but has failed to stop it. This is very likely to create a memory leak. 

异常信息显示: 应用程序启动了一个名为pool-5-thread-2的线程,但是关闭失败,这很可能导致内存泄漏。

很明显,tomcat没能关掉ThreadPoolExecutor的核心线程,因此需要在关闭tomcat前手动关闭。

如何手动关闭线程池请参考:停止Tomcat webapp 线程池报错的尝试解决

那如果我不想手动关闭线程池怎么办?

使用Spring中中的 ThreadPoolTaskExecutor。将线程池的关闭交给Spring去管理。

在applicationContext.xml中加入

 <!-- spring thread pool executor -->
    <bean id="taskExecutor" class="org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor">
        <!-- 线程池维护线程的最少数量 -->
        <property name="corePoolSize" value="10"/>
        <!-- 允许的空闲时间 -->
        <property name="keepAliveSeconds" value="5"/>
        <!-- 线程池维护线程的最大数量 -->
        <property name="maxPoolSize" value="20"/>
        <!-- 缓存队列 -->
        <property name="queueCapacity" value="4"/>
        <!-- 对拒绝task的处理策略 -->
        <property name="rejectedExecutionHandler">
            <bean class="java.util.concurrent.ThreadPoolExecutor$CallerRunsPolicy"/>
        </property>
    </bean>

程序中:

    @Autowired
    private ThreadPoolTaskExecutor threadPool;

     threadPool.execute(() -> {
            model.addAttribute("types", typeService.findHottestType(6));
            countDownLatch.countDown();
        });
     threadPool.execute(() -> {
            model.addAttribute("tags", tagService.findHottestTag(6));
            countDownLatch.countDown();
        });
     threadPool.execute(() -> {
            model.addAttribute("blogs", blogService.findHotestBlogs());
            countDownLatch.countDown();
        });
     threadPool.execute(() -> {
            model.addAttribute("newBlogs", blogService.findNewestBlogs(8));
            countDownLatch.countDown();
        });

再次启动程序,测试,关闭tomcat,报错信息不再提示。

 

如何优雅安全的关闭线程池?

线程池关闭不合理,导致应用⽆法正常stop的情况,这是我们很容易忽略的。

下面就以ThreadPoolExecutor为例,来介绍下如何优雅的关闭线程池。

线程中断

 

在介绍线程池关闭之前,先介绍下Thread的interrupt。

在程序中,我们是不能随便中断⼀个线程的,因为这是极其不安全的操作,我们⽆法知道这个线程正运⾏在什么状态,它可能持有某把锁,强⾏中断可能导致锁不能释放的问题;或者线程可能在操作数据库,强⾏中断导致数据不一致,从而混乱的问题。正因此,Java⾥将Thread的stop⽅法设置为过时,以禁⽌⼤家使⽤。

⼀个线程什么时候可以退出呢?当然只有线程自⼰才能知道。

所以我们这⾥要说的Thread的interrrupt⽅法,本质不是⽤来中断一个线程。而是将线程设置⼀个中断状态。当我们调⽤线程的interrupt方法,它有两个作⽤用:

1、如果此线程处于阻塞状态(比如调⽤了wait方法,io等待),则会立刻退出阻塞,并抛出InterruptedException异常,线程就可以通过捕获InterruptedException来做⼀定的处理,然后让线程退出。

2、如果此线程正处于运行之中,则线程不受任何影响,继续运行,仅仅是线程的中断标记被设置为true。所以线程要在适当的位置通过调用isInterrupted方法来查看自⼰是否被中断,并做退出操作。

注:如果线程的interrupt方法先被调用,然后线程调用阻塞方法进入阻塞状态,InterruptedException异常依旧会抛出。如果线程捕获InterruptedException异常后,继续调用阻塞方法, 将不再触发InterruptedException异常。

线程池的关闭

 

线程池提供了两个关闭方法,shutdownNow和shuwdown⽅法。

shutdownNow⽅法的解释是:线程池拒接收新提交的任务,同时⽴刻关闭线程池,线程池里的任务不再执行

shutdown⽅法的解释是:线程池拒接收新提交的任务,同时等待线程池⾥的任务执行完毕后关闭线程池。

以上的说法虽然没错,但是还有很多的细节问题,比如调用shutdown⽅法后,正在执⾏的任务的线程会做出什么反应?正在等待任务的线程又会做出什么反应?线程在什么情况下才会彻底退出。如果不了解这些细节,在关闭线程池时就难免遇到,像线程池关闭不了,关闭线程池出现报错等情况。

再说这些关闭线程池细节问题之前,需要强调一点的是,调用完shutdownNow和shuwdown⽅法后,并不代表线程池已经完成关闭操作,它只是异步的通知线程池进行关闭处理。如果要同步等待线程池彻底关闭后才继续往下执行,需要调⽤awaitTermination⽅法进⾏同步等待。

 

我们来看看shutdownNow和shuwdown⽅法的源码:

shutdown:

/**
     * Initiates an orderly shutdown in which previously submitted
     * tasks are executed, but no new tasks will be accepted.
     * Invocation has no additional effect if already shut down.
     *
     * <p>This method does not wait for previously submitted tasks to
     * complete execution.  Use {@link #awaitTermination awaitTermination}
     * to do that.
     *
     * @throws SecurityException {@inheritDoc}
     */
   public void shutdown() {
    final ReentrantLock mainLock = this.mainLock;
    // 上锁,确保同一时间只能有一个线程执行此操作
    mainLock.lock();
    try {
        // 检查方法调用方是否用权限关闭线程池以及中断工作线程
        checkShutdownAccess();
        // 将线程池运行状态设置为SHUTDOWN
        advanceRunState(SHUTDOWN);
        // 中断所有空闲线程
        interruptIdleWorkers();
        // 此方法在ThreadPoolExecutor中是空实现,具体实现在其子类ScheduledThreadPoolExecutor
        // 中,用于取消延时任务。
        onShutdown(); // hook for ScheduledThreadPoolExecutor
    } finally {
        mainLock.unlock();
    }
    // 尝试将线程池置为TERMINATED状态
    tryTerminate();
}

 

 

 

 

 

 

shutdownNow:

    /**
     * Attempts to stop all actively executing tasks, halts the
     * processing of waiting tasks, and returns a list of the tasks
     * that were awaiting execution. These tasks are drained (removed)
     * from the task queue upon return from this method.
     *
     * <p>This method does not wait for actively executing tasks to
     * terminate.  Use {@link #awaitTermination awaitTermination} to
     * do that.
     *
     * <p>There are no guarantees beyond best-effort attempts to stop
     * processing actively executing tasks.  This implementation
     * cancels tasks via {@link Thread#interrupt}, so any task that
     * fails to respond to interrupts may never terminate.
     *
     * @throws SecurityException {@inheritDoc}
     */
public List<Runnable> shutdownNow() {
    List<Runnable> tasks;
    final ReentrantLock mainLock = this.mainLock;
    mainLock.lock();
    try {
        // 检查方法调用方是否用权限关闭线程池以及中断工作线程
        checkShutdownAccess();
        // 将线程池运行状态置为STOP
        advanceRunState(STOP);
        // 中断所有线程,包括正在运行的线程
        interruptWorkers();
        // 将未执行的任务移入列表中
        tasks = drainQueue();
    } finally {
        mainLock.unlock();
    }
    // 尝试将线程池置为TERMINATED状态
    tryTerminate();
    return tasks;
}

具体源码解释请参考:关闭线程池的正确姿势

 

 

 

 

 

 

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 如果需要关闭某个线程,可以通过中断(interrupt)线程的方式终止它的执行。示例代码如下: ```java ExecutorService executor = Executors.newFixedThreadPool(10); Future<?> future = executor.submit(() -> { // 异步任务的代码 }); // 中断正在执行的任务 future.cancel(true); ``` 在调用submit()方法后,返回一个Future对象,表示异步任务的执行结果。可以通过调用该对象的cancel()方法来中断任务的执行。如果传入的参数为true,则尝试中断任务的执行,否则等待任务执行完成。 需要注意的是,中断任务的执行并不立即生效,而是需要在异步任务中自行判断线程的中断状态,然后在合适的时机终止任务的执行。示例代码如下: ```java ExecutorService executor = Executors.newFixedThreadPool(10); Future<?> future = executor.submit(() -> { while (!Thread.currentThread().isInterrupted()) { // 异步任务的代码 } }); // 中断正在执行的任务 future.cancel(true); ``` 在异步任务中,可以通过判断Thread.currentThread().isInterrupted()的返回值来判断线程是否被中断。如果返回true,则表示线程被中断,需要终止任务的执行。在任务的代码中,可以通过抛出InterruptedException异常来终止任务的执行,示例代码如下: ```java ExecutorService executor = Executors.newFixedThreadPool(10); Future<?> future = executor.submit(() -> { try { while (!Thread.currentThread().isInterrupted()) { // 异步任务的代码 } } catch (InterruptedException e) { // 终止任务的执行 } }); // 中断正在执行的任务 future.cancel(true); ``` ### 回答2: 在使用Java异步服务时,线程池的正确关闭是非常重要的。如果线程池不能正常关闭,将导致线程暴增,从而导致系统资源耗尽,甚至宕机。以下是一个示例代码,展示了如何正确关闭线程池。 ```java import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; public class AsynchronousServiceExample { private ExecutorService threadPool; public void startService() { // 初始化线程池 threadPool = Executors.newFixedThreadPool(10); // 启动异步任务 for (int i = 0; i < 100; i++) { threadPool.submit(() -> { // 异步任务的具体逻辑 // ... }); } } public void stopService() { // 关闭线程池并等待任务执行完成 threadPool.shutdown(); // 请求线程池关闭 try { if (!threadPool.awaitTermination(60, TimeUnit.SECONDS)) { // 等待线程池中的任务执行完毕,最多等待60秒 threadPool.shutdownNow(); // 强制关闭线程池 if (!threadPool.awaitTermination(60, TimeUnit.SECONDS)) { // 如果线程池仍未关闭,则输出错误日志 System.err.println("线程池未能正常关闭"); } } } catch (InterruptedException e) { // 捕获线程中断异常 threadPool.shutdownNow(); // 如果线程等待时被中断,则强制关闭线程池 Thread.currentThread().interrupt();// 重新设置线程中断状态 } } } ``` 在上述示例代码中,我们首先利用`Executors.newFixedThreadPool()`方法创建线程池,大小为10,并启动了100个异步任务。当需要关闭线程池时,调用`threadPool.shutdown()`方法来请求线程池关闭。然后,使用`threadPool.awaitTermination()`方法等待线程池中的任务执行完成,最多等待60秒。如果超过60秒仍有任务没有执行完毕,使用`threadPool.shutdownNow()`方法强制关闭线程池。最后,通过判断线程池是否成功关闭来决定是否输出错误日志。为了处理线程中断异常,我们在`awaitTermination()`方法和`shutdownNow()`方法之后重新设置了线程中断状态,以确保程序的稳定性。 ### 回答3: 当使用线程池时,可能遇到线程暴增的问题。为了解决这个问题,可以采取以下步骤来关闭线程: 步骤1: 停止向线程池提交新的任务。这可以通过调用线程池的shutdown方法来完成,该方法将不再接受新任务,并尝试将所有已提交但未执行的任务完成。 步骤2: 等待线程池中的任务执行完成。可以使用awaitTermination方法来等待,该方法等待指定时间,直到所有任务都执行完成或者超时。 步骤3: 如果在等待时间内,线程池中的任务仍未执行完成,可以使用shutdownNow方法来中止所有未执行的任务。该方法发送中断信号给线程池中的线程,使其尽快停止执行。 下面是一个示例代码,演示了如何使用线程池来执行异步任务,并关闭线程池: ```java import java.util.concurrent.ExecutorService; import java.util.concurrent.Executors; import java.util.concurrent.TimeUnit; public class ThreadPoolExample { public static void main(String[] args) { // 创建线程池,指定最大线程数 ExecutorService executor = Executors.newFixedThreadPool(10); // 提交任务给线程池执行 for (int i = 0; i < 100; i++) { executor.execute(new Worker()); } // 停止向线程池提交新的任务 executor.shutdown(); try { // 等待线程池中的任务执行完成,最多等待1小时 if (!executor.awaitTermination(1, TimeUnit.HOURS)) { // 在等待时间内,任务未执行完成,进行中断处理 executor.shutdownNow(); // 等待线程池中的任务中止 if (!executor.awaitTermination(1, TimeUnit.MINUTES)) { System.err.println("线程池未能完全停止"); } } } catch (InterruptedException e) { // 发生中断异常,进行中断处理 executor.shutdownNow(); } } private static class Worker implements Runnable { @Override public void run() { // 执行具体的任务逻辑 // ... } } } ``` 上述示例代码中,首先创建了一个固定大小的线程池,然后使用execute方法提交任务给线程池执行。接着调用shutdown方法停止向线程池提交新的任务,然后使用awaitTermination方法等待线程池中的任务执行完成。如果等待时间超过指定的超时时间(在示例中为1小时),则调用shutdownNow方法中止未执行的任务。最后,再次等待线程池中的任务中止,如果超时,则打印一条警告信息。如果在任何等待期间发生了InterruptedException异常,则调用shutdownNow方法中止未执行的任务。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值