CountDownLatch--等待多线程计数器

一、概述

CountDownLatch是什么?网上概念的描述太多了。其实个人理解,他就是--等待多线程计数器

即:在某线程中(主线程或其他分线程都可以)声明一个初始值为N的CountDownLatch计数器,然后等待多个子线程完成了相关操作后再继续向下执行。--当各子线程执行到相应的地方后使N-1,最后N=0时,线程不再等待,继续向下执行。

 

二、运用及场景

2.1、运用

  1. CountDownLatch latch = new CountDownLatch(N); //构造对象时候 需要传入参数N  即计数器值
  2. latch.await();//能够阻塞线程 直到调用N次latch.countDown() 方法才释放线程
  3. latch.countDown();//在多个子线程中调用 每次调用,N计数器值-1

2.2、场景

需要等待其他线程做完某动作后再继续执行后续操作(当前线程多异步操作完成后再向下执行)--或等待超时后继续执行。
例:每天0点同时备份前一天的数据表,即0点开启与备份表相同的线程数分别同时备份表,全部成功后返回true,失败后返回false

 

 

三、案例解析

3.1、描述一个案例:3个人在等公交,公交停下后,分别上车,找到座位后车才启动。

先创建一个值为3(nameList.length)的CountDownLatch计数器 → 启动3个子线程分别执行任务(之后,主线程陷入等待latch.await()) → 子线程执行后将计数器-1(latch.countDown()) → 计数器值为0后唤醒主线程继续向下执行

public static void main(String[] args) {
    try {
        Random random = new Random();//生成随机数类
	SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");//设置日期格式
	System.out.println("现在是北京时间:"+df.format(new Date())+"======>公交车停车了!");
		 
	String[] nameList = new String[]{"张三","李四","王五"};//假设此为在公交站台排队的人
	CountDownLatch latch = new CountDownLatch(nameList.length);//声明一个与人数相等的计数器
	for(String name : nameList){
	    new Thread(new Runnable() {
	    @Override
	    public void run() {
	        try {
		    System.out.println(name+"小老弟,上车了!===》"+df.format(new Date()));
		    int s = random.nextInt(6);//生成0-6的随机数,用做等待秒数
		    System.out.println(name+"小老弟,找座位需要===》"+s+"秒!");
		    Thread.sleep(s*1000);//等待s秒
		    System.out.println(name+"小老弟,找到座位了!===》"+df.format(new Date()));
		} catch (Exception e) {
		    e.printStackTrace();
		} finally {
		    latch.countDown();//计数器-1
		}
	    }
	    }).start();
	}
	latch.await();//等待
	System.out.println("现在是北京时间:"+df.format(new Date())+"======>公交车启动了!");
    } catch (InterruptedException e) {
	e.printStackTrace();
    }
}

 

结果:

 

3.2、描述一个案例:3个人在等公交,公交停下后分别上车,待2人找到位置后,或2秒之后,公交启动。

先创建一个值为2的CountDownLatch计数器 → 启动3个子线程分别执行任务(之后,主线程陷入等待latch.await(2, TimeUnit.SECONDS),超时时间设置为2秒) → 子线程执行后将计数器-1(latch.countDown()) → 计数器值为0或等待超时后 唤醒主线程继续向下执行

public static void main(String[] args) {
    try {
        Random random = new Random();//生成随机数类
	SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");//设置日期格式
	System.out.println("现在是北京时间:"+df.format(new Date())+"======>公交车停车了!");
		 
	String[] nameList = new String[]{"张三","李四","王五"};//假设此为在公交站台排队的人
	CountDownLatch latch = new CountDownLatch(2);//声明一个与人数相等的计数器
	for(String name : nameList){
	    new Thread(new Runnable() {
	    @Override
	    public void run() {
	        try {
		    System.out.println(name+"小老弟,上车了!===》"+df.format(new Date()));
		    int s = random.nextInt(6);//生成0-6的随机数,用做等待秒数
		    System.out.println(name+"小老弟,找座位需要===》"+s+"秒!");
		    Thread.sleep(s*1000);//等待s秒
		    System.out.println(name+"小老弟,找到座位了!===》"+df.format(new Date()));
		} catch (Exception e) {
		    e.printStackTrace();
		} finally {
	            latch.countDown();//计数器-1
		}
	    }
	    }).start();
	}
	latch.await(2, TimeUnit.SECONDS);//等待计数器为0,或2秒
	System.out.println("现在是北京时间:"+df.format(new Date())+"======>公交车启动了!");
    } catch (InterruptedException e) {
        e.printStackTrace();
    }
}

结果:

可能性1:

可能性2:

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章