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

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个子线程已经执行完毕,继续执行主线程



你可能感兴趣的:(3.ITOO)