java线程间通讯的一些方法总结

前言

并发编程中,我们可能会遇到这样一个场景

A、B两个线程并行,但是我希望保证B线程在A线程执行完了后再执行

这个时候就需要线程间进行通讯

A执行完了后对B说一声,喂B,我执行完了

来康康用Java怎么实现

1、基于synchronized

2、基于reentrantLock

3、基于volatile

4、基于countDownLatch

我目前就知道这四种

1、synchronized+wait() 和 notify()

wait() 和 notify()都是Object类的通讯方法,注意一点,wait和 notify必须搭配synchronized使用,并且wait()会释放锁,notify()不会释放锁

public class SynchronizedTest {

 //定义个year,用来记录某明星的练习年数
 private static double year;

 public void run() {
  //线程A,练习唱跳rap
  Thread threadA = new Thread(() -> {
   synchronized (this) {
    for (year = 0.5; year <= 5; year += 0.5) {
     System.out.println("蔡徐鸡开始练习唱跳rap:已练习" + year + "年");
     try {
      Thread.sleep(288);
     } catch (InterruptedException e) {
      e.printStackTrace();
     }
     //众所周知,练习两年半即可出道
     if (year == 2.5) {
      System.out.println("===========================>成功练习两年半,出道!!!");
      this.notify();
     }
    }
   }
  });
  //线程B,练习打篮球
  Thread threadB = new Thread(() -> {
   while (true) {
    synchronized (this) {
     try {
      this.wait();
     } catch (InterruptedException e) {
      e.printStackTrace();
     }
     System.out.println("蔡徐鸡开始练习打篮球");
    }
   }
  });
  //注意,一定要先启动B,不然会导致B永远拿不到锁
  threadB.start();
  threadA.start();
 }

 public static void main(String[] args) {
  SynchronizedTest test = new SynchronizedTest();
  test.run();
 }
}

运行结果:

蔡徐鸡开始练习唱跳rap:已练习0.5年
蔡徐鸡开始练习唱跳rap:已练习1.0年
蔡徐鸡开始练习唱跳rap:已练习1.5年
蔡徐鸡开始练习唱跳rap:已练习2.0年
蔡徐鸡开始练习唱跳rap:已练习2.5年
===========================>成功练习两年半,出道!!!
蔡徐鸡开始练习唱跳rap:已练习3.0年
蔡徐鸡开始练习唱跳rap:已练习3.5年
蔡徐鸡开始练习唱跳rap:已练习4.0年
蔡徐鸡开始练习唱跳rap:已练习4.5年
蔡徐鸡开始练习唱跳rap:已练习5.0年
蔡徐鸡开始练习打篮球

注意看运行结果,线程A在执行notify后并没有释放锁,而是执行完当前任务才开始执行线程B的任务

2、基于ReentrantLock

ReentrantLock也能实现线程间通讯,不过有点麻烦,需要结合ReentrantLock的Condition

public class LockTest {
  //定义个year,用来记录某明星练习打篮球的年数
  private static double year;

  public static void main(String[] args) {
   ReentrantLock lock = new ReentrantLock();
   Condition condition = lock.newCondition();
   //线程A,练习唱跳rap
   Thread threadA = new Thread(() -> {
    //执行业务代码前上锁
    lock.lock();
    for (year = 0.5; year <= 5; year += 0.5) {
     System.out.println("蔡徐鸡开始练习唱跳rap:已练习" + year + "年");
     try {
      Thread.sleep(288);
     } catch (InterruptedException e) {
      e.printStackTrace();
     }
     //众所周知,练习两年半即可出道
     if (year == 2.5) {
      System.out.println("===========================>成功练习两年半,出道!!!");
      //唤醒等待中的线程
      condition.signal();
     }
    }
    //业务代码执行完后解锁
    lock.unlock();
   });
   //线程B,练习打篮球
   Thread threadB = new Thread(() -> {
    //执行业务代码前上锁
    lock.lock();
    while (true) {
     try {
      //让线程等待,如果计数器为0的话,则立即执行
      condition.await();
     } catch (InterruptedException e) {
      e.printStackTrace();
     }
     System.out.println("蔡徐老母鸡开始练习打篮球");
     break;
    }
    //业务代码执行完后解锁
    lock.unlock();
   });
   //注意,一定要先启动B,不然会导致B永远拿不到锁
   threadB.start();
   threadA.start();
  }
 }

运行结果:

蔡徐鸡开始练习唱跳rap:已练习0.5年
蔡徐鸡开始练习唱跳rap:已练习1.0年
蔡徐鸡开始练习唱跳rap:已练习1.5年
蔡徐鸡开始练习唱跳rap:已练习2.0年
蔡徐鸡开始练习唱跳rap:已练习2.5年
===========================>成功练习两年半,出道!!!
蔡徐鸡开始练习唱跳rap:已练习3.0年
蔡徐鸡开始练习唱跳rap:已练习3.5年
蔡徐鸡开始练习唱跳rap:已练习4.0年
蔡徐鸡开始练习唱跳rap:已练习4.5年
蔡徐鸡开始练习唱跳rap:已练习5.0年
蔡徐老母鸡开始练习打篮球

效果和synchronized+wait() 和 notify()一样一样的

3、基于volatile

使用共享变量也能实现,用volatile即可,原理就是多个线程共同监听同个变量,根据变量的值变化来执行对应的任务,此处volatile的作用就是让其它线程能即时感知变量值的改变

public class volatileTest {
 //定义一个共享变量,注意,必须用volatile修饰
 static volatile boolean flag = false;
 //定义个year,用来记录某明星练习打篮球的年数
 private static double year;

 public static void main(String[] args) {
  //线程A,练习唱跳rap
  Thread threadA = new Thread(() -> {
   while (true) {
    if (!flag) {
     for (year = 0.5; year <= 5; year += 0.5) {
      System.out.println("蔡徐鸡开始练习唱跳rap:已练习" + year + "年");
      try {
       Thread.sleep(288);
      } catch (InterruptedException e) {
       e.printStackTrace();
      }
      //众所周知,练习两年半即可出道
      if (year == 2.5) {
       System.out.println("===========================>成功练习两年半,出道!!!");
       year = 0.5;
       flag = true;
       break;
      }
     }
    }
   }
  });
  //线程B,练习打篮球
  Thread threadB = new Thread(() -> {
   while (true) {
    if (flag) {
     System.out.println("蔡徐老母鸡开始练习打篮球");
     break;
    }
   }
  });
  // 启动线程
  threadA.start();
  threadB.start();
 }
}

运行结果:

蔡徐鸡开始练习唱跳rap:已练习0.5年
蔡徐鸡开始练习唱跳rap:已练习1.0年
蔡徐鸡开始练习唱跳rap:已练习1.5年
蔡徐鸡开始练习唱跳rap:已练习2.0年
蔡徐鸡开始练习唱跳rap:已练习2.5年
===========================>成功练习两年半,出道!!!
蔡徐老母鸡开始练习打篮球

基于CountDownLatch

CountDownLatch是JUC包下的一个并发编程工具,主要有两个方法,countDown和await,CountDownLatch底层维护了一个计数器,在实例化的时候设置,当调用countDown方法时,计数器减一,如果计数器在减一前已经为0,那么什么都不会发生,如果减一后变成0,则唤醒所有等待的线程;await方法会使当前线程等待,直到计数器为0

public class CountDownLatchTest {
 //定义个year,用来记录某明星练习打篮球的年数
 private static double year;

 public static void main(String[] args) {
  CountDownLatch latch = new CountDownLatch(1);
  //线程A,练习唱跳rap
  Thread threadA = new Thread(() -> {
   for (year = 0.5; year <= 5; year += 0.5) {
    System.out.println("蔡徐鸡开始练习唱跳rap:已练习" + year + "年");
    try {
     Thread.sleep(288);
    } catch (InterruptedException e) {
     e.printStackTrace();
    }
    //众所周知,练习两年半即可出道
    if (year == 2.5) {
     System.out.println("===========================>成功练习两年半,出道!!!");
     //计数器减一
     latch.countDown();
    }
   }
  });
  //线程B,练习打篮球
  Thread threadB = new Thread(() -> {
   while (true) {
    try {
     //让线程等待,如果计数器为0的话,则立即执行
     latch.await();
    } catch (InterruptedException e) {
     e.printStackTrace();
    }
    System.out.println("蔡徐老母鸡开始练习打篮球");
    break;
   }
  });
  // 启动线程
  threadA.start();
  threadB.start();
 }
}

运行结果:

蔡徐鸡开始练习唱跳rap:已练习0.5年
蔡徐鸡开始练习唱跳rap:已练习1.0年
蔡徐鸡开始练习唱跳rap:已练习1.5年
蔡徐鸡开始练习唱跳rap:已练习2.0年
蔡徐鸡开始练习唱跳rap:已练习2.5年
===========================>成功练习两年半,出道!!!
蔡徐鸡开始练习唱跳rap:已练习3.0年
蔡徐老母鸡开始练习打篮球
蔡徐鸡开始练习唱跳rap:已练习3.5年
蔡徐鸡开始练习唱跳rap:已练习4.0年
蔡徐鸡开始练习唱跳rap:已练习4.5年
蔡徐鸡开始练习唱跳rap:已练习5.0年

如果你多运行几次,你会发现线程B执行的时机是随机的,但永远在计数器为0后才开始执行,也就是说计数器为0后,线程A和线程B谁抢到锁就谁执行

文中所有demo都是复制即可运行,大家还是要多动手,家里有条件的都用idea跑一跑,没条件的可以用手抄

总结

到此这篇关于java线程间通讯的文章就介绍到这了,更多相关java线程间通讯内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Java并发编程线程间通讯实现过程详解

    在Java中线程间通讯有多种方式,我这里列出一些常用方式,并用代码的方式展示他们是如何实现的: 共享变量 wait, notify,notifyAll(这3个方法是Object对象中的方法,且必须与synchronized关键字结合使用) CyclicBarrier.CountDownLatch 利用LockSupport Lock/Condition机制 管道,创建管道输出流PipedOutputStream和管道输入流PipedInputStream 示例一: package com.zhi

  • 浅谈Java多线程实现及同步互斥通讯

    Java多线程深入理解本文主要从三个方面了解和掌握多线程: 1. 多线程的实现方式,通过继承Thread类和通过实现Runnable接口的方式以及异同点. 2. 多线程的同步与互斥中synchronized的使用方法. 3. 多线程的通讯中的notify(),notifyAll(),及wait(),的使用方法,以及简单的生成者和消费者的代码实现. 下面来具体的讲解Java中的多线程: 一:多线程的实现方式 通过继承Threa类来实现多线程主要分为以下三步: 第一步:继承 Thread,实现Thr

  • JAVA多线程间通讯常用实现方法解析

    如何实现线程间通讯,有如下三种方法: 1.使用Semaphore (信号量)类来控制线程的等待和释放 功能:三个线程 a .b .c 并发运行,b,c 需要 a 线程的数据怎么实现 分析:考虑到多线程的不确定性, 因此我们不能确保 ThreadA 就一定先于 ThreadB 和 ThreadC 前执行,就算 ThreadA先执行了, 我们也无法保证 ThreadA 什么时候才能将变量 num 给初始化完成. 因此我们必须让 ThreadB 和 Thread去等待 ThreadA 完成任何后发出的

  • Java多线程通讯之wait,notify的区别详解

    下面通过代码给大家介绍java多线程通讯之wait notify的区别,具体内容如下所示: class Res{ public String username; public String sex; } class Out extends Thread{ Res res; public Out(Res res){ this.res=res; } @Override public void run() { //写操作 int count=0; while (true){ // synchroniz

  • java线程间通讯的一些方法总结

    前言 并发编程中,我们可能会遇到这样一个场景 A.B两个线程并行,但是我希望保证B线程在A线程执行完了后再执行 这个时候就需要线程间进行通讯 A执行完了后对B说一声,喂B,我执行完了 来康康用Java怎么实现 1.基于synchronized 2.基于reentrantLock 3.基于volatile 4.基于countDownLatch 我目前就知道这四种 1.synchronized+wait() 和 notify() wait() 和 notify()都是Object类的通讯方法,注意一

  • 浅谈Java线程间通信之wait/notify

    Java中的wait/notify/notifyAll可用来实现线程间通信,是Object类的方法,这三个方法都是native方法,是平台相关的,常用来实现生产者/消费者模式.先来我们来看下相关定义: wait() :调用该方法的线程进入WATTING状态,只有等待另外线程的通知或中断才会返回,调用wait()方法后,会释放对象的锁. wait(long):超时等待最多long毫秒,如果没有通知就超时返回. notify() :通知一个在对象上等待的线程,使其从wait()方法返回,而返回的前提

  • java线程间通信的通俗解释及代码示例

    线程间通信:由于多线程共享地址空间和数据空间,所以多个线程间的通信是一个线程的数据可以直接提供给其他线程使用,而不必通过操作系统(也就是内核的调度). 进程间的通信则不同,它的数据空间的独立性决定了它的通信相对比较复杂,需要通过操作系统.以前进程间的通信只能是单机版的,现在操作系统都继承了基于套接字(socket)的进程间的通信机制.这样进程间的通信就不局限于单台计算机了,实现了网络通信.线程通信主要分为以下几个部分,下面通过生活中图书馆借书的例子简单讲解以下: 通过共享对象通信 加入图书馆只有

  • Java线程中start和run方法全面解析

    自定义线程两种方法 自定义一个runnable接口的实现类,然后构造一个thread,即对thread传入一个runnable接口类. new一个thread或者写个thread子类,覆盖它的run方法.(new 一个thread并覆盖run方法实际上是匿名内部类的一种方式) 示例代码 public static void main(String[] args) { new Thread(new Runnable() { @Override public void run() { System.

  • Java线程间通信不同步问题原理与模拟实例

    本文实例讲述了Java线程间通信不同步问题原理与模拟.分享给大家供大家参考,具体如下: 一 点睛 下面两种情况可造成线程间不同步: 1 生产者没生产完,消费者就来消费. 2 消费者没消费完,生产者又来生产,覆盖了还没来得及消费的数据. 二 代码 class Producer implements Runnable { private Person person = null; public Producer( Person person ) { this.person = person; } @

  • Java线程死锁实例及解决方法

    这篇文章主要介绍了Java线程死锁实例及解决方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 1.死锁的定义 所谓死锁是指多个线程因竞争资源而造成的一种僵局(互相等待),若无外力作用,这些进程都将无法向前推进 2.死锁产生的必要条件 互斥条件:线程要求对所分配的资源(如打印机)进行排他性控制,即在一段时间内某资源仅为一个线程所占有.此时若有线程请求该资源,则请求线程只能等待. 不剥夺条件:线程所获得的资源在未使用完毕之前,不能被其他线程倾向夺

  • 创建Java线程安全类的七种方法

    目录 前言 无状态 没有共享状态 消息传递 不可变状态 使用来自 java.util.concurrent 的数据结构 同步块 易失性领域 总结 前言 几乎每个 Java 应用程序都使用线程.像 Tomcat 这样的 Web 服务器在单独的工作线程中处理每个请求,胖客户端在专用工作线程中处理长时间运行的请求,甚至批处理使用 java.util.concurrent.ForkJoinPool 来提高性能. 因此,有必要以线程安全的方式编写类,这可以通过以下技术之一来实现. 无状态 当多个线程访问同

  • Java线程间共享实现方法详解

    一.synchronize对象锁和类锁 synchronize为多线程关键字是一种同步锁,它可以修饰以下几种对象: 代码块:被修饰的代码块被称为同步代码块,作用的范围是{}里面的代码,作用的对象是调用这个代码块的对象 方法:被修饰的方法称为同步方法,作用的范围是整个方法,作用的对象是调用这个方法的对象 类:作用的范围是synchronize后面括号里的部分,作用的对象是当前这个类 1.对象锁 下面由一个栗子引入: public class TestSynchronize { //加了对象锁的方法

随机推荐