一、概述
CountDownLatch是什么?网上概念的描述太多了。其实个人理解,他就是--等待多线程计数器!
即:在某线程中(主线程或其他分线程都可以)声明一个初始值为N的CountDownLatch计数器,然后等待多个子线程完成了相关操作后再继续向下执行。--当各子线程执行到相应的地方后使N-1,最后N=0时,线程不再等待,继续向下执行。
二、运用及场景
2.1、运用
- CountDownLatch latch = new CountDownLatch(N); //构造对象时候 需要传入参数N 即计数器值
- latch.await();//能够阻塞线程 直到调用N次latch.countDown() 方法才释放线程
- 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: