Java_判断线程池所有任务是否执行完毕

import java.util.concurrent.ExecutorService;  
import java.util.concurrent.Executors;  
  
public class Test {  
    public static void main(String args[]) throws InterruptedException {  
        ExecutorService exe = Executors.newFixedThreadPool(50);  
        for (int i = 1; i <= 5; i++) {  
            exe.execute(new SubThread(i));  
        }  
        exe.shutdown();  
        while (true) {  
            if (exe.isTerminated()) {  
                System.out.println("结束了!");  
                break;  
            }  
            Thread.sleep(200);  
        }  
    }  
}

1、使用exe.shutdown()和exe.isTerminated();判断线程结束

上面是主线程的代码,创建了一个能同时执行2个线程的线程池,并投入5个线程,当5个线程都执行完毕后打印---“结束了!”字符串。

exe.shutdown();该方法在加入线程队列的线程执行完之前不会执行。

exe.isTerminated();当shutdown()或者shutdownNow()执行了之后才会执行,并返回true。

在上面的代码中必须有exe.isTerminated()的判断,否则在投入5个线程到线程池后会直接打印:“结束了”。不能达到我们想要的效果。

通过while(true)循环判断exe.isTerminated()重生之大文豪的值,为了防止过多的判断浪费资源,可设置线程睡眠Thread.sleep(200);

正是由于这个睡眠,所以当所有线程池中的线程都执行完后,有可能延迟200ms才执行"结束了"语句。这个参数越小延迟越小,结果越准确。

下面是子线程,子线程只是简单的将数字i打印出来;

public class SubThread extends Thread{  
    private final int i;  
    public SubThread(int i){  
        this.i = i;  
    }  
    @Override  
    public void run(){  
        System.out.println(i);  
    }  
}
执行结果:

run:  3  1 4  5  2  结束了!  

成功构建 (总时间: 2 秒)  

子线程执行顺序不能控制,所以输出的结果是乱序的。

2 使用Thread的join()等待所有的子线程执行完毕,主线程在执行

import java.util.Vector;

public class ThreadSubMain2 {
    public static void main(String[] args) {
        Vector<Thread> threads = new Vector<Thread>(); // 使用线程安全的Vector
        for (int i = 0; i < 10; i++) {
            Thread iThread = new Thread(new Runnable() {
                public void run() {
                    try {
                        Thread.sleep(1000); // 模拟子线程任务
                    } catch (InterruptedException e) {
                    }
                    System.out.println("子线程" + Thread.currentThread() + "执行完毕");
                }
            });
            threads.add(iThread);
            iThread.start();
        }
        for (Thread iThread : threads) {
            try { // 等待所有线程执行完毕
                iThread.join();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
        System.out.println("主线执行。");
    }
}

执行结果如下:

子线程Thread[Thread-1,5,main]执行完毕

子线程Thread[Thread-2,5,main]执行完毕

子线程Thread[Thread-0,5,main]执行完毕

子线程Thread[Thread-3,5,main]执行完毕

子线程Thread[Thread-4,5,main]执行完毕

子线程Thread[Thread-9,5,main]执行完毕

子线程Thread[Thread-7,5,main]执行完毕

子线程Thread[Thread-5,5,main]执行完毕

子线程Thread[Thread-8,5,main]执行完毕

子线程Thread[Thread-6,5,main]执行完毕

主线执行。

这种方式符合要求,它能够等待所有的子线程执行完,主线程才会执行。

3 使用 ExecutorService 线程池,等待所有任务执行完毕再执行主线程, awaitTermination 。

awaitTermination(long timeout,TimeUnit unit)

请求关闭、发生超时或者当前线程中断,无论哪一个首先发生之后,都将导致阻塞,直到所有任务完成执行。

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;

public class ThreadSubMain3 {
    public static void main(String[] args) {
        ExecutorService threadPool = Executors.newCachedThreadPool(); // 定义一个缓冲的线程值 线程池的大小根据任务变化
        for (int i = 0; i < 10; i++) {
            threadPool.execute(new Runnable() {
                public void run() {
                    try {
                        Thread.sleep(1000); // 模拟子线程任务
                    } catch (InterruptedException e) {
                    }
                    System.out.println("子线程" + Thread.currentThread() + "执行完毕");
                }
            });
        }
        threadPool.shutdown(); // 启动一次顺序关闭,执行以前提交的任务,但不接受新任务。
        try {
            // 请求关闭、发生超时或者当前线程中断,无论哪一个首先发生之后,都将导致阻塞,直到所有任务完成执行
            // 设置最长等待10秒
            threadPool.awaitTermination(10, TimeUnit.SECONDS);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("主线执行。");
    }
}

执行结果如下:

子线程Thread[pool-1-thread-4,5,main]执行完毕

子线程Thread[pool-1-thread-1,5,main]执行完毕

子线程Thread[pool-1-thread-7,5,main]执行完毕

子线程Thread[pool-1-thread-6,5,main]执行完毕

子线程Thread[pool-1-thread-5,5,main]执行完毕

子线程Thread[pool-1-thread-2,5,main]执行完毕

子线程Thread[pool-1-thread-3,5,main]执行完毕

子线程Thread[pool-1-thread-8,5,main]执行完毕

子线程Thread[pool-1-thread-10,5,main]执行完毕

子线程Thread[pool-1-thread-9,5,main]执行完毕

主线执行。

这种方法和方法2一样,将等待所有子线程执行完毕之后才执行主线程。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值