JAVA线程池shutdown和shutdownNow的区别


         shutDown()

    当线程池调用该方法时,线程池的状态则立刻变成SHUTDOWN状态。此时,则不能再往线程池中添加任何任务,否则将会抛出RejectedExecutionException异常。但是,此时线程池不会立刻退出,直到添加到线程池中的任务都已经处理完成,才会退出。

            shutdownNow()

     根据JDK文档描述,大致意思是:执行该方法,线程池的状态立刻变成STOP状态,并试图停止所有正在执行的线程,不再处理还在池队列中等待的任务,当然,它会返回那些未执行的任务。
     它试图终止线程的方法是通过调用Thread.interrupt()方法来实现的,但是大家知道,这种方法的作用有限,如果线程中没有sleep 、wait、Condition、定时锁等应用, interrupt()方法是无法中断当前的线程的。所以,ShutdownNow()并不代表线程池就一定立即就能退出,它可能必须要等待所有正在执行的任务都执行完成了才能退出。

    上面对shutDown()以及shutDownNow()作了一个简单的、理论上的分析。如果想知道why,则需要亲自打开JDK源码,分析分析。
      想要分析shutDown()以及shutDownNow()源码,我建议首先要对ThreadPoolExecutor有个大概了解。因为关闭线程池的所有方法逻辑都在ThreadPoolExecutor中处理的。
      如果你真的想知道为什么,建议看一下我以前写的一篇对ThreadPoolExecutor源码分析的博文,我想这对你比较透彻的了解shutDown()和shutDownNow()的区别以及java 线程池原理有很大的帮助。博文URL:
         http://xtu-xiaoxin.iteye.com/admin/blogs/647744
     废话少说,要查看源码,首先进入ThreadPoolExecutor的shutDown()方法:

  

Java代码  

java线程执行完再往下走 java线程shutdown_java

java线程执行完再往下走 java线程shutdown_java线程执行完再往下走_02


1. public void
2.      SecurityManager security = System.getSecurityManager();  
3. if (security != null)  
4.             security.checkPermission(shutdownPerm);  
5. final ReentrantLock mainLock = this.mainLock;  
6.         mainLock.lock();  
7. try
8. if (security != null) { // Check if caller can modify our threads
9. for
10.                     security.checkAccess(w.thread);  
11.             }  
12. int
13. if
14. //设置线程池状态为关闭状态
15. //----------------代码1
16. try
17. for
18. //一个一个中断线程
19. //-----------------代码2
20.                 }  
21. catch (SecurityException se) { // Try to back out
22.                 runState = state;  
23. // tryTerminate() here would be a no-op
24. throw
25.             }  
26. // Terminate now if pool and queue empty
27. finally
28.             mainLock.unlock();  
29.         }  
30.     }


public void shutdown() {
	 SecurityManager security = System.getSecurityManager();
	if (security != null)
            security.checkPermission(shutdownPerm);
        final ReentrantLock mainLock = this.mainLock;
        mainLock.lock();
        try {
            if (security != null) { // Check if caller can modify our threads
                for (Worker w : workers)
                    security.checkAccess(w.thread);
            }
            int state = runState;
            if (state < SHUTDOWN)
            	//设置线程池状态为关闭状态
                runState = SHUTDOWN;     //----------------代码1
            try {
                for (Worker w : workers) {
                	//一个一个中断线程
                    w.interruptIfIdle();  //-----------------代码2
                }
            } catch (SecurityException se) { // Try to back out
                runState = state;
                // tryTerminate() here would be a no-op
                throw se;
            }
            tryTerminate(); // Terminate now if pool and queue empty
        } finally {
            mainLock.unlock();
        }
    }




 

看上面源码,代码1是线程池关闭的关键,如果线程池状态一旦设为SHUTDOWN,则在线程池中会出现两种现象:

     1.你不能再往线程池中添加任何任务,否则会抛RejectedExecutionException异常(详细请看ThreadPoolExecutor的addIfUnderCorePoolSize方法)。

     2.工作线程Worker获得池队列中的任务时(详细看Worker中的getTask()方法)的处理逻辑也发生了变化:如果线程池为RUNNING状态,并且池队列中没任务时,它会一直等待,直到你提交任务到池队列中,然后取出任务,返回。但是,一旦你执行了shutDown()方法,线程池状态为SHUTDOWN状态,它将不再等待了,直接返回null。如果返回null,则工作线程没有要执行的任务,直接退出(详细看Worker中run()方法)。


    代码2是针对这种情况的:在线程池关闭前,有部分工作线程就一直在等着要处理的任务,也就是说工作线程空闲着(这种情况我描述的不好,其实就是Worker正在执行getTask()方法中’ r = workQueue.take();’代码段)。这时,调用interrupt()方法来中断这些Worker线程。进入代码2看看吧:。


  


 


Java代码  

java线程执行完再往下走 java线程shutdown_java

java线程执行完再往下走 java线程shutdown_java线程执行完再往下走_02



    1. void
    2. final ReentrantLock runLock = this.runLock;  
    3. /*
    4.              * 注意这个条件,摆明的就是要等Worker中runTask()方法运行完后才成立。
    5.              * 锁机制
    6.              */
    7. if
    8. try
    9. /*
    10.              * 如果当前工作线程没有正在运行,则中断线程
    11.              * 他能中断工作线程的原因是getTask()方法能抛出一个
    12.              * InterruptedException。这时,则可终止那些正在执行
    13.              * workQueue.take()方法的工作线程
    14.              */
    15. if
    16.             thread.interrupt();           
    17. finally
    18.                     runLock.unlock();  
    19.                 }  
    20.             }  
    21.         }



    void interruptIfIdle() {
                final ReentrantLock runLock = this.runLock;
                /*
                 * 注意这个条件,摆明的就是要等Worker中runTask()方法运行完后才成立。
                 * 锁机制
                 */
                if (runLock.tryLock()) {
                    try {
                /*
                 * 如果当前工作线程没有正在运行,则中断线程
                 * 他能中断工作线程的原因是getTask()方法能抛出一个
                 * InterruptedException。这时,则可终止那些正在执行
                 * workQueue.take()方法的工作线程
                 */
    		    if (thread != Thread.currentThread())
    			thread.interrupt();         
                    } finally {
                        runLock.unlock();
                    }
                }
            }




      

    最后进入shutDownNow()方法看看,这个更简单了,就是设置线程池状态为STOP,然后依次调用工作线程的interrupt()方法,就这么简单,最后还是把源码贴出来吧:

       


         


    Java代码  

    java线程执行完再往下走 java线程shutdown_java

    java线程执行完再往下走 java线程shutdown_java线程执行完再往下走_02



      1. public
      2. /*
      3.         * shutdownNow differs from shutdown only in that
      4.         * 1. runState is set to STOP,
      5.         * 2. all worker threads are interrupted, not just the idle ones, and
      6.         * 3. the queue is drained and returned.
      7.         */
      8. SecurityManager security = System.getSecurityManager();  
      9. if (security != null)  
      10.            security.checkPermission(shutdownPerm);  
      11.   
      12. final ReentrantLock mainLock = this.mainLock;  
      13.        mainLock.lock();  
      14. try
      15. if (security != null) { // Check if caller can modify our threads
      16. for
      17.                    security.checkAccess(w.thread);  
      18.            }  
      19.   
      20. int
      21. if
      22.                runState = STOP;  
      23.   
      24. try
      25. for
      26.                    w.interruptNow();  
      27.                }  
      28. catch (SecurityException se) { // Try to back out
      29.                runState = state;  
      30. // tryTerminate() here would be a no-op
      31. throw
      32.            }  
      33.   
      34.            List<Runnable> tasks = drainQueue();  
      35. // Terminate now if pool and queue empty
      36. return
      37. finally
      38.            mainLock.unlock();  
      39.        }  
      40.    }



      public List<Runnable> shutdownNow() {
              /*
               * shutdownNow differs from shutdown only in that
               * 1. runState is set to STOP,
               * 2. all worker threads are interrupted, not just the idle ones, and
               * 3. the queue is drained and returned.
               */
      	SecurityManager security = System.getSecurityManager();
      	if (security != null)
                  security.checkPermission(shutdownPerm);
      
              final ReentrantLock mainLock = this.mainLock;
              mainLock.lock();
              try {
                  if (security != null) { // Check if caller can modify our threads
                      for (Worker w : workers)
                          security.checkAccess(w.thread);
                  }
      
                  int state = runState;
                  if (state < STOP)
                      runState = STOP;
      
                  try {
                      for (Worker w : workers) {
                          w.interruptNow();
                      }
                  } catch (SecurityException se) { // Try to back out
                      runState = state;
                      // tryTerminate() here would be a no-op
                      throw se;
                  }
      
                  List<Runnable> tasks = drainQueue();
                  tryTerminate(); // Terminate now if pool and queue empty
                  return tasks;
              } finally {
                  mainLock.unlock();
              }
          }



         

      上面代码没什么好分析的了,一看就明白,其实别看上面代码一大篇,我们只关心“w.interruptNow();”即可。

            还是那句话,希望对需要了解的人有点帮助。