今天学习CountDownLatch这个类,作用感觉和join很像,然后就百度了一下,看了他们之间的区别。所以在此记录一下。
首先来看一下join,在当前线程中,如果调用某个thread的join方法,那么当前线程就会被阻塞,直到thread线程执行完毕,当前线程才能继续执行。join的原理是,不断的检查thread是否存活,如果存活,那么让当前线程一直wait,直到thread线程终止,线程的this.notifyAll 就会被调用。
我们来看一下这个应用场景:假设现在公司有三个员工A,B,C,他们要开会。但是A需要等B,C准备好之后再才能开始,B,C需要同时准备。我们先用join模拟上面的场景。
Employee.java:
public class Employee extends Thread{ private String employeeName; private long time; public Employee(String employeeName,long time){ this.employeeName = employeeName; this.time = time; } @Override public void run() { try { System.out.println(employeeName+ "开始准备"); Thread.sleep(time); System.out.println(employeeName+" 准备完成"); } catch (Exception e) { e.printStackTrace(); } } }
JoinTest.java:
public class JoinTest { public static void main(String[] args) throws InterruptedException { Employee a = new Employee("A", 3000); Employee b = new Employee("B", 3000); Employee c = new Employee("C", 4000); b.start(); c.start(); b.join(); c.join(); System.out.println("B,C准备完成"); a.start(); } }
最后输出结果如下:
C开始准备 B开始准备 B 准备完成 C 准备完成 B,C准备完成 A开始准备 A 准备完成
可以看到,A总是在B,C准备完成之后才开始执行的。
CountDownLatch中我们主要用到两个方法一个是await()方法,调用这个方法的线程会被阻塞,另外一个是countDown()方法,调用这个方法会使计数器减一,当计数器的值为0时,因调用await()方法被阻塞的线程会被唤醒,继续执行。
接下来,我们用CountDownLatch来模拟一下。
Employee.java:
public class Employee extends Thread{ private String employeeName; private long time; private CountDownLatch countDownLatch; public Employee(String employeeName,long time, CountDownLatch countDownLatch){ this.employeeName = employeeName; this.time = time; this.countDownLatch = countDownLatch; } @Override public void run() { try { System.out.println(employeeName+ "开始准备"); Thread.sleep(time); System.out.println(employeeName+" 准备完成"); countDownLatch.countDown(); } catch (Exception e) { e.printStackTrace(); } } }
CountDownLatchTest.java:
public class CountDownLatchTest { public static void main(String[] args) throws InterruptedException { CountDownLatch countDownLatch = new CountDownLatch(2); Employee a = new Employee("A", 3000,countDownLatch); Employee b = new Employee("B", 3000,countDownLatch); Employee c = new Employee("C", 4000,countDownLatch); b.start(); c.start(); countDownLatch.await(); System.out.println("B,C准备完成"); a.start(); } }
输出结果如下:
B开始准备 C开始准备 B 准备完成 C 准备完成 B,C准备完成 A开始准备 A 准备完成
上面可以看到,CountDownLatch与join都能够模拟上述的场景,那么他们有什么不同呢?这时候我们试想另外一个场景就能看到他们的区别了。
假设A,B,C的工作都分为两个阶段,A只需要等待B,C各自完成他们工作的第一个阶段就可以执行了。
我们来修改一下Employee类:
public class Employee extends Thread{ private String employeeName; private long time; private CountDownLatch countDownLatch; public Employee(String employeeName,long time, CountDownLatch countDownLatch){ this.employeeName = employeeName; this.time = time; this.countDownLatch = countDownLatch; } @Override public void run() { try { System.out.println(employeeName+ " 第一阶段开始准备"); Thread.sleep(time); System.out.println(employeeName+" 第一阶段准备完成"); countDownLatch.countDown(); System.out.println(employeeName+ " 第二阶段开始准备"); Thread.sleep(time); System.out.println(employeeName+" 第二阶段准备完成"); } catch (Exception e) { e.printStackTrace(); } } }
CountDownLatchTest类不需要做修改,输出结果入下:
B 第一阶段开始准备 C 第一阶段开始准备 B 第一阶段准备完成 B 第二阶段开始准备 C 第一阶段准备完成 C 第二阶段开始准备 B,C第一阶段准备完成 A 第一阶段开始准备 B 第二阶段准备完成 A 第一阶段准备完成 A 第二阶段开始准备 C 第二阶段准备完成 A 第二阶段准备完成
从结果可以看出,A在B,C第一阶段准备完成的时候就开始执行了,不需要等到第二阶段准备完成。这种场景下,用join是没法实现的。
总结:调用join方法需要等待thread执行完毕才能继续向下执行,而CountDownLatch只需要检查计数器的值为零就可以继续向下执行,相比之下,CountDownLatch更加灵活一些,可以实现一些更加复杂的业务场景。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持小牛知识库。
在Kotlin中是否有任何特定的语言实现,它与其他语言实现的Coroutines有什么不同? 什么意思是coroutine就像轻量线程? 有什么区别? kotlin余式实际上是并行/并发运行的吗? 即使在多核系统中,在任何给定的时间也只有一个协调线在运行(对吗?) 在这里,我开始100000余次,这段代码后面发生了什么?
本文向大家介绍inner join和left join之间的区别详解,包括了inner join和left join之间的区别详解的使用技巧和注意事项,需要的朋友参考一下 前言 关于inner join 与 left join 之间的区别,以前以为自己搞懂了,今天从前端取参数的时候发现不是预想中的结果,才知道问题出在inner join 上了。 需求是从数据库查数据,在前端以柱形图的形式展现出来,
本文向大家介绍详解Java中Thread 和Runnable区别,包括了详解Java中Thread 和Runnable区别的使用技巧和注意事项,需要的朋友参考一下 Thread 和Runnable 关系 Thread类是接口Runnable的一个实现类。 源码分析 Thread Threa类运行的时候调用start()方法,源代码如下: 调用start()方法,实际运行的是start0方法,方法声
问题内容: 这两个连接将给我相同的结果: 与 语句之间的性能或其他方面有什么区别吗? 不同的 SQL 实现之间是否有所不同? 问题答案: 它们在功能上是等效的,但阅读起来可能更清晰,尤其是在查询中包含其他联接类型(即or或)的情况下。
本文向大家介绍SQL联合查询inner join、outer join和cross join的区别详解,包括了SQL联合查询inner join、outer join和cross join的区别详解的使用技巧和注意事项,需要的朋友参考一下 对于开发使用到数据库的应用,免不了就要使用联合查询,SQL中常用的联合查询有inner join、outer join和cross join;这三者的区别很多人
本文向大家介绍浅析Java中Runnable和Thread的区别,包括了浅析Java中Runnable和Thread的区别的使用技巧和注意事项,需要的朋友参考一下 线程的起动并不是简单的调用了你的RUN方法,而是由一个线程调度器来分别调用你的所有线程的RUN方法, 我们普通的RUN方法如果没有执行完是不会返回的,也就是会一直执行下去,这样RUN方法下面的方法就不可能会执行了,可是线程里的RUN方法