JAVA CountDownLatch与thread-join()的区别解析

 更新时间:2019年08月06日 09:42:14   作者:卑微幻想家  
这篇文章主要介绍了JAVA CountDownLatch与thread-join()的区别解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下

今天学习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更加灵活一些,可以实现一些更加复杂的业务场景。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • java 中Comparable与Comparator详解与比较

    java 中Comparable与Comparator详解与比较

    这篇文章主要介绍了java 中Comparable与Comparator详解与比较的相关资料,需要的朋友可以参考下
    2017-04-04
  • Logback日志存放路径不统一解决方案

    Logback日志存放路径不统一解决方案

    这篇文章主要介绍了Logback日志存放路径不统一解决方案,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-10-10
  • Java7到Java17之Switch语句进化史示例详解

    Java7到Java17之Switch语句进化史示例详解

    这篇文章主要为大家介绍了Java7到Java17之Switch语句进化史示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-01-01
  • Spring boot整合ELK详细过程

    Spring boot整合ELK详细过程

    ELK是由Elasticsearch、Logstash和Kibana三个开源软件组成的技术堆栈,主要用于数据的存储、处理和可视化,本文给大家介绍Spring boot整合ELK详细过程,感兴趣的朋友一起看看吧
    2024-01-01
  • JDBC使用Statement修改数据库

    JDBC使用Statement修改数据库

    这篇文章主要为大家详细介绍了JDBC使用Statement修改数据库,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-08-08
  • Java 生成任意长度的验证码过程解析

    Java 生成任意长度的验证码过程解析

    这篇文章主要介绍了Java 生成任意长度的验证码过程解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2019-10-10
  • java高级排序之希尔排序

    java高级排序之希尔排序

    这篇文章主要介绍了java高级排序之希尔排序 ,需要的朋友可以参考下
    2015-04-04
  • 关于JwtToken使用-重点看一下过期时间

    关于JwtToken使用-重点看一下过期时间

    这篇文章主要介绍了关于JwtToken使用-重点看一下过期时间,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-07-07
  • Mybatis-Plus可能导致死锁的问题分析及解决办法

    Mybatis-Plus可能导致死锁的问题分析及解决办法

    这篇文章给大家主要介绍了Mybatis-Plus可能导致死锁的问题分析及解决办法,文中通过代码示例给大家介绍的非常详细,具有一定的参考价值,需要的朋友可以参考下
    2023-12-12
  • Spring IOC容器的Bean管理基于注解属性注入方式

    Spring IOC容器的Bean管理基于注解属性注入方式

    这篇文章主要为大家介绍了Spring IOC容器的Bean管理基于注解属性注入方式,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-05-05

最新评论