Java多线程通信lock和wait

在Java多线程中有一对配合使用过的两个方法,来实现线程间通信的功能--lock和wait, 由于这个需要获得锁,所以必须结合synchronized一起使用。首先我们先看一个例子:


public class LockWait {
  
  static volatile List<String> itemContainer = new ArrayList<>();
  static Object obj = new Object();
  
  public static void main(String[] args) {
      Thread th1 = new Thread(() -> {
          synchronized (obj) {
              for (int i = 0; i < 10; i++) {
                  System.out.println("th1添加元素");
                  itemContainer.add(String.valueOf(i));
                  if (itemContainer.size() == 5) {
                      System.out.println("th1线程发出通知");
                      obj.notify();
                  }
              }
          }
      });
      
      Thread th2 = new Thread(() -> {
          synchronized (obj) {
              System.out.println("进入th2线程");
              if (itemContainer.size() != 5) {
                  try {
                      System.out.println("th2线程开始等待");
                      obj.wait();
                      System.out.println("th2线程等待结束");
                  } catch (InterruptedException e) {
                      e.printStackTrace();
                  }
                  System.out.println("th2线程结束");
              }
          }
          
      });
      
      th2.start();
      th1.start();
  }
}

输出结果如下:

进入th2线程
th2线程开始等待
th1添加元素
th1添加元素
th1添加元素
th1添加元素
th1添加元素
th1线程发出通知
th1添加元素
th1添加元素
th1添加元素
th1添加元素
th1添加元素
th2线程等待结束
th2线程结束

具体运行逻辑如下:

01.png

总结上面的运行结果,th2在wait的时候,th1可以持有锁。说明wait是释放锁,而notify不释放锁。

这样也就带来了一个弊端,无法实时的得到结果,就是说当List达到我们想要的结果的时候,th1线程一直还在持有锁,导致th2无法执行。

有没有更好办法呢?在Java中提供了一个CountDownLatch类:


public class CountDownLatchTest {
  
  public static void main(String[] args) {
      final List<String> itemContainer = new ArrayList<>();
      final CountDownLatch countDownLanch = new CountDownLatch(1);
      Thread th1 = new Thread(() -> {
          for (int i = 0; i < 10; i++) {
              try {
                  System.out.println("th1添加元素");
                  itemContainer.add(String.valueOf(i));
                  if (itemContainer.size() == 5) {
                      System.out.println("th1线程发出通知");
                      countDownLanch.countDown();
                  }
              } catch (Exception e) {
                  e.printStackTrace();
              }
              
          }
      });
      
      Thread th2 = new Thread(() -> {
          System.out.println("进入th2线程");
          if (itemContainer.size() != 5) {
              try {
                  System.out.println("th2线程开始等待");
                  countDownLanch.await();
                  System.out.println("th2线程等待结束");
              } catch (InterruptedException e) {
                  e.printStackTrace();
              }
              System.out.println("th2线程结束");
          }
          
      });
      
      th2.start();
      
      th1.start();
  }
}



运行结果:

进入th2线程
th1添加元素
th2线程开始等待
th1添加元素
th1添加元素
th1添加元素
th1添加元素
th1线程发出通知
th1添加元素
th2线程等待结束
th1添加元素
th2线程结束
th1添加元素
th1添加元素
th1添加元素

gzh.jpg

获得最新的博主文章,请关注上方公众号

©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Java-Review-Note——4.多线程 标签: JavaStudy PS:本来是分开三篇的,后来想想还是整...
    coder_pig阅读 1,672评论 2 17
  • Java多线程学习 [-] 一扩展javalangThread类 二实现javalangRunnable接口 三T...
    影驰阅读 2,986评论 1 18
  • 本文主要讲了java中多线程的使用方法、线程同步、线程数据传递、线程状态及相应的一些线程函数用法、概述等。 首先讲...
    李欣阳阅读 2,482评论 1 15
  • 线程池ThreadPoolExecutor corepoolsize:核心池的大小,默认情况下,在创建了线程池之后...
    irckwk1阅读 756评论 0 0
  • 整理图片的时候,发现了这张多年前存下来的一首小诗。 四月是我最爱的月份,多年从未改变。它没有三月的微寒,也没有五月...
    迪迪_L阅读 173评论 0 1