via: http://blog.csdn.net/truong/article/details/40227435
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);
}
}
}
上面是主线程的代码,创建了一个能同时执行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 秒)
子线程执行顺序不能控制,所以输出的结果是乱序的。
via: http://www.tuicool.com/articles/ZvAFny
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一样,将等待所有子线程执行完毕之后才执行主线程。
相关推荐
Java 判断线程池所有任务是否执行完毕的操作 在 Java 中,判断线程池所有任务是否执行完毕是非常重要的操作。线程池是一种高效的并发编程方式,但是在使用线程池时,需要确保所有任务都已经执行完毕,以避免系统...
Java判断线程池线程是否执行完毕是Java多线程编程中的一种常见问题,当我们使用线程池来执行多个任务时,需要判断所有的子线程是否已经执行完毕,以便进行后续操作。下面我们将介绍两种解决方案:使用Semaphore和...
SHUTDOWN状态,此时线程池不能够接受新的任务,它会等待所有任务执行完毕 static final int STOP = 2;停止状态;此时线程池不能接受新的任务,并且会去尝试终止正在执行的任务 static final int TERMINATED = 3;终止...
6. 关闭线程池:所有任务执行完毕后,关闭线程池,释放资源。 在压缩包文件“线程池D7”中,可能包含了`uThreadPool`的源代码、示例程序、以及运行和测试所需的相关文件。通过查看这些文件,我们可以更深入地理解`...
JAVA 主线程等待子线程执行完毕再执行 JAVA 中的线程控制是非常重要的一部分,而在实际开发中,我们经常会遇到需要主线程等待子线程执行完毕再执行的情况。这种情况下,我们可以使用两种方式来实现:主动式和被动式...
然而,如何确保主线程在所有子线程执行完毕后再进行后续操作,成为一个挑战。 首先,我们看到代码中创建了多个线程并发执行任务。每个线程由一个Runnable对象表示,这些Runnable对象在run()方法中执行实际的数据...
4. `isDownloadFinished()`:判断当前块是否已下载完毕,用于同步和通知主类`DownLoad`。 在实际操作中,`DownLoad`类可能会创建多个`DownLoadTask`实例,将它们提交到ExecutorService执行,然后监听所有任务的完成...
shutdown() 方法会正常关闭线程池,等待所有任务执行完毕后关闭,而 shutdownNow() 方法会强行关闭线程池,取消所有正在执行的任务。 使用 shutdown() 方法来停止线程服务 ----------------------------- shutdown...
`join()`方法确保主线程等待所有子线程执行完毕后再继续。 在实际的多线程编程中,我们还需要考虑线程安全问题,避免数据竞争和死锁。此外,线程优先级、线程池等也是提高系统性能的重要手段。对于更复杂的场景,还...
- **Future和Callable**:Callable接口的实现类可以返回结果,Future接口代表Callable的执行结果,可以获取结果或判断任务是否完成。 7. **线程优先级** - Java提供10个线程优先级,从MIN_PRIORITY(1)到MAX_...
- **死亡状态**(Terminated):执行完毕或异常终止。 #### 3. 创建线程的两种方式 - **继承Thread类**:重写`run()`方法。 - **实现Runnable接口**:实现`run()`方法,并通过`Thread(Runnable target)`构造函数...
- **Terminated**:线程池完全终止,terminated()方法执行完毕。 3. **JDK、JRE、JVM的区别**: - **JDK**:包含编译器javac、JRE以及开发所需的类库,用于开发Java应用程序。 - **JRE**:Java运行时环境,包含...
- **终止(Terminated)**:`run()`方法执行完毕或抛出未捕获异常。 3. **线程控制** - **线程同步**:包括`synchronized`关键字、`wait()`/`notify()`/`notifyAll()`方法,用于解决多线程环境下的数据一致性问题...
- **终止(Terminated)**:run()方法执行完毕或抛出异常,线程结束。 3. **线程同步与互斥** - **synchronized**:通过关键字synchronized实现临界区保护,确保同一时刻只有一个线程访问共享资源。 - **...
- 第二线程执行完毕后再次调用countDown,计数器减1。 - 第三线程等待计数器变为0后开始执行。 - **使用CyclicBarrier**: - 创建CyclicBarrier对象,屏障数设为3。 - 三个线程分别到达屏障后,都会等待其他线程...
在Java中,判断线程是否已经死亡,可以使用`Thread.isAlive()`方法。如果线程处于运行、就绪或阻塞状态,这个方法返回`true`;反之,如果线程处于新建或死亡状态,它将返回`false`。例如,在上述的`ThreadTest`示例...
在Java编程中,多线程是一个重要的概念,它允许程序同时执行多个任务,提高程序的效率和响应速度。在给定的练习题中,主要涉及了两个知识点:Java中的多线程实现以及线程同步。 首先,我们来看第一部分的代码。这是...
最后,等待所有线程执行完毕,收集并分析结果。 以下是一个简单的多线程ping Python代码示例: ```python import os import threading def ping_ip(ip): result = os.system('ping -c 1 ' + ip) # 在这里替换为...