Java并发实战:CountDownLatch实现多线程抽题

时间:2021-10-16 17:58:15

1.定义

CountDownLatch这个类能够使一个线程等待其他线程完成各自的工作后再执行。CountDownLatch是通过一个计数器来实现的,计数器的初始值为线程的数量。每当一个线程完成了自己的任务后,计数器的值就会减1。当计数器值到达0时,它表示所有的线程已经完成了任务,然后在闭锁上等待的线程就可以恢复执行任务。



2.区别CyclicBarrier

  • CountDownLatch和CyclicBarrier都能够实现线程之间的等待,只不过它们侧重点不同:
  • CountDownLatch一般用于某个线程A等待若干个其他线程执行完任务之后,它才执行;
  • 而CyclicBarrier一般用于一组线程互相等待至某个状态,然后这一组线程再同时执行;
  • 另外,CountDownLatch是不能够重用的,而CyclicBarrier是可以重用的。


3.主要方法


countDown:主线程必须在启动其他线程后立即调用CountDownLatch.await()方法。这样主线程的操作就会在这个方法上阻塞,直到其他线程完成各自的任务。

await:其他N 个线程必须引用闭锁对象,因为他们需要通知CountDownLatch对象,他们已经完成了各自的任务。这种通知机制是通过 CountDownLatch.countDown()方法来完成的;每调用一次这个方法,在构造函数中初始化的count值就减1。所以当N个线程都调 用了这个方法,count的值等于0,然后主线程就能通过await()方法,恢复执行自己的任务。


4.应用场景

  • 实现最大的并行性:有时我们想同时启动多个线程,实现最大程度的并行性。
  • 开始执行前等待n个线程完成各自任务:例如应用程序启动类要确保在处理用户请求前,所有N个外部系统已经启动和运行了。
  • 死锁检测:一个非常方便的使用场景是,你可以使用n个线程访问共享资源,在每次测试阶段的线程数目是不同的,并尝试产生死锁。

5.在登陆后抽题的过程中使用的等待n个线程完成各自任务。

package test.pool;
import java.util.concurrent.CountDownLatch;


public class testCountDownLatch {
public static void main(String[] args) {
final CountDownLatch countDownLatch=new CountDownLatch(4);
new Thread(){
public void run(){
try {
System.out.println(Thread.currentThread().getName()+"抽取听力题");
Thread.sleep(3000);
countDownLatch.countDown();
} catch (Exception e) {
e.printStackTrace();
}
}
}.start();

new Thread(){
public void run(){
try {
System.out.println(Thread.currentThread().getName()+"抽取单选题");
Thread.sleep(3000);
countDownLatch.countDown();
} catch (Exception e) {
e.printStackTrace();
}
}
}.start();

new Thread(){
public void run(){
try {
System.out.println(Thread.currentThread().getName()+"抽取复合式题");
Thread.sleep(3000);
countDownLatch.countDown();
} catch (Exception e) {
e.printStackTrace();
}
}
}.start();

new Thread(){
public void run(){
try {
System.out.println(Thread.currentThread().getName()+"抽取作文题");
Thread.sleep(3000);
countDownLatch.countDown();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}.start();


try {
Thread.sleep(1000);
System.out.println("4个线程正在抽题呢");
countDownLatch.await();
System.out.println("4个子线程已经执行完毕,继续执行主线程");

} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

console:

Thread-0抽取听力题Thread-1抽取单选题Thread-2抽取复合式题Thread-3抽取作文题4个线程正在抽题呢4个子线程已经执行完毕,继续执行主线程