http://blog.chenlb.com/2008/12/main-thread-wait-all-sub-thread-finish-task-in-thread-pool.html

原文出处:http://blog.chenlb.com/2008/12/main-thread-wait-all-sub-thread-finish-task-in-thread-pool.html

用线程池编写多线程程序时,当所有任务完成时,要做一些统计的工作。而统计工作必须要在所有任务完成才能做。所以要让主线程等待所有任务完成。可以使用ThreadPoolExecutor.awaitTermination(long timeout, TimeUnit unit)。请看示例代码:

  1. package com.chenlb;
  2. import java.util.Random;
  3. import java.util.concurrent.LinkedBlockingQueue;
  4. import java.util.concurrent.ThreadPoolExecutor;
  5. import java.util.concurrent.TimeUnit;
  6. /**
  7. * 线程池使用示例, 主线程等待所有任务完成再结束.
  8. *
  9. * @author chenlb 2008-12-2 上午10:31:03
  10. */
  11. public class ThreadPoolUse {
  12. public static class MyTask implements Runnable {
  13. private static int id = 0;
  14. private String name = "task-"+(++id);
  15. private int sleep;
  16. public MyTask(int sleep) {
  17. super();
  18. this.sleep = sleep;
  19. }
  20. public void run() {
  21. System.out.println(name+" -----start-----");
  22. try {
  23. Thread.sleep(sleep);    //模拟任务执行.
  24. } catch (InterruptedException e) {
  25. e.printStackTrace();
  26. }
  27. System.out.println(name+" -----end "+sleep+"-----");
  28. }
  29. }
  30. public static void main(String[] args) {
  31. System.out.println("==================start==================");
  32. ThreadPoolExecutor executor = new ThreadPoolExecutor(5,5, 60, TimeUnit.SECONDS, new LinkedBlockingQueue<Runnable>());
  33. int n = 10;
  34. int sleep = 10 * 1000;  //10s
  35. Random rm = new Random();
  36. for(int i=0; i<n; i++) {
  37. executor.execute(new MyTask(rm.nextInt(sleep)+1));
  38. }
  39. executor.shutdown();//只是不能再提交新任务,等待执行的任务不受影响
  40. try {
  41. boolean loop = true;
  42. do {    //等待所有任务完成
  43. loop = !executor.awaitTermination(2, TimeUnit.SECONDS);  //阻塞,直到线程池里所有任务结束
  44. } while(loop);
  45. } catch (InterruptedException e) {
  46. e.printStackTrace();
  47. }
  48. System.out.println("==================end====================");
  49. }
  50. }

当然还有其它方法。

http://xtu-xiaoxin.iteye.com/blog/649677

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代码  
  1. public void shutdown() {
  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 (Worker w : workers)
  10. security.checkAccess(w.thread);
  11. }
  12. int state = runState;
  13. if (state < SHUTDOWN)
  14. //设置线程池状态为关闭状态
  15. runState = SHUTDOWN;     //----------------代码1
  16. try {
  17. for (Worker w : workers) {
  18. //一个一个中断线程
  19. w.interruptIfIdle();  //-----------------代码2
  20. }
  21. } catch (SecurityException se) { // Try to back out
  22. runState = state;
  23. // tryTerminate() here would be a no-op
  24. throw se;
  25. }
  26. tryTerminate(); // Terminate now if pool and queue empty
  27. } finally {
  28. mainLock.unlock();
  29. }
  30. }

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

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

Java代码  
  1. public List<Runnable> shutdownNow() {
  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. final ReentrantLock mainLock = this.mainLock;
  12. mainLock.lock();
  13. try {
  14. if (security != null) { // Check if caller can modify our threads
  15. for (Worker w : workers)
  16. security.checkAccess(w.thread);
  17. }
  18. int state = runState;
  19. if (state < STOP)
  20. runState = STOP;
  21. try {
  22. for (Worker w : workers) {
  23. w.interruptNow();
  24. }
  25. } catch (SecurityException se) { // Try to back out
  26. runState = state;
  27. // tryTerminate() here would be a no-op
  28. throw se;
  29. }
  30. List<Runnable> tasks = drainQueue();
  31. tryTerminate(); // Terminate now if pool and queue empty
  32. return tasks;
  33. } finally {
  34. mainLock.unlock();
  35. }
  36. }

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

主线程等待线程池所有任务完成相关推荐

  1. Java主线程等待子线程、线程池

    public class TestThread extends Thread { public void run() { System.out.println(this.getName() + &qu ...

  2. java等待5秒_Java并发编程-主线程等待子线程解决方案

    主线程等待所有子线程执行完成之后,再继续往下执行的解决方案 public class TestThread extends Thread { public void run() { System.ou ...

  3. 【多线程】学习记录七种主线程等待子线程结束之后在执行的方法

    最近遇到一个问题需要主线程等待所有的子线程结束,才能开始执行,统计所有的子线程执行结果,返回,网上翻阅各种资料,最后记录一下,找到七种方案 第一种:while循环 对于"等待所有的子线程结束 ...

  4. 面试官:如何让主线程等待所有的子线程执行结束之后再执行

    java 主线程等待所有子线程执行完毕在执行,在工作总往往会遇到异步去执行某段逻辑, 然后先处理其他事情, 处理完后再把那段逻辑的处理结果进行汇总(比如用户下单一个产品,后台会做一系列的处理,为了提高 ...

  5. java主线程等待所有子线程执行完毕再执行

    java主线程等待所有子线程执行完毕在执行,这个需求其实我们在工作中经常会用到,比如用户下单一个产品,后台会做一系列的处理,为了提高效率,每个处理都可以用一个线程来执行,所有处理完成了之后才会返回给用 ...

  6. 如何实现java主线程等待子线程执行完毕之后再执行?

    本文转自:问题:如何实现java主线程等待子线程执行完毕之后再执行? - jseven - 博客园 点击关注强哥,查看更多精彩文章呀 工作总往往会遇到异步去执行某段逻辑, 然后先处理其他事情, 处理完 ...

  7. 主线程等待所有子线程结束的4种方法

    目录 主线程不等待子线程全部结束 1.使用CountDownLatch 2.同步屏障CyclicBarrier 2.1.CyclicBarrier使用 2.2.CyclicBarrier复用 2.3. ...

  8. 【Java并发编程】主线程等待子线程的多种方法

    文章目录 1.Thread sleep() 2.Thread join() 3.synchronized 等待唤醒机制 4.ExecutorService isTerminated() + while ...

  9. c++主线程等待子线程结束_简单明了的 Python 多线程来了 | 原力计划

    作者 | 万里羊责编 | 王晓曼出品 | CSDN博客线程和进程计算机的核心是CPU,它承担了所有的计算任务,就像是一座工厂在时刻运行.如果工厂的资源有限,一次只能供一个车间来使用,也就是说当一个车间 ...

最新文章

  1. R语言数值累加函数cumsum实战
  2. NMS和roi pooling 实现以及加速
  3. ATO/MTO类机械制造业特点以及ERP需求分析(三)
  4. 那些年帮助我们理解各种姿势、常见协议的大神动图
  5. Eclipse安装Gradle插件及配置
  6. Java工具类——通过配置XML验证Map
  7. Spring Data JPA: 实现自定义Repository
  8. jude的一些基本用法
  9. C# 使用AutoResetEvent进行线程同步
  10. Unix/Linux Command Reference
  11. php判断手机是安卓系统还是ios系统
  12. Centos7中完美搭建ftp服务器
  13. 图文详细解说DevExpress 2015新版亮点【附文档下载】
  14. java flv 转swf_swf转flv格式转换器 使用ffmpeg进行视频文件转换成FLV整理
  15. DXF解析CAD图形解析PLT格式文件解析C#工程源码
  16. 一开电脑wifi就断网_为什么电脑一连wifi就断网,
  17. java毕业设计城市猎人户外军品店Mybatis+系统+数据库+调试部署
  18. URL中的20%、22%、26%、7B%、%7D、28%、29%解析成真实的字符
  19. 小程序中如何正确使用换行符‘\n‘
  20. 网络协议与服务的区别/关系

热门文章

  1. android 录屏自动运行,app自动化--Android通过adb录屏
  2. Redis的安装以及基本操作简介
  3. 【小白学习PyTorch教程】十九、 基于torch实现UNet 图像分割模型
  4. 深度学习和目标检测系列教程 11-300:小麦数据集训练Faster-RCNN模型
  5. 物理化学 热力学第一定律的概念
  6. 漫谈度量学习(Distance Metric Learning)那些事儿
  7. 直播 | 北邮博士生纪厚业:异质图神经网络在阿里推荐业务中的探索
  8. 论文落地 101:算法工程化的那些坑
  9. java实验四 集合与函数式编程实验
  10. 求取给定的二叉树的镜像_17---二叉树的镜像