Java——JUC高并发编程,面试必问( 线程间通信 )

3.线程间通信

线程间通信的模型有两种:共享内存和消息传递,以下方式都是基本这两种模型来实现的。我们来基本一道面试常见的题目来分析:

面试题

场景—两个线程,一个线程对当前数值加 1,另一个线程对当前数值减 1,要求用线程间通信。

3.1 synchronized 方案

/**
* volatile 关键字实现线程交替加减
*/
public class TestVolatile {
 /**
 * 交替加减 	
 */
   public static void main(String[] args){
      DemoClass demoClass = new DemoClass();
      new Thread(() ->{
           for (int i = 0; i < 5; i++) { d
              emoClass.increment();
      }
   }, "线程 A").start();
   
       new Thread(() ->{
            for (int i = 0; i < 5; i++) {
              demoClass.decrement();
            }
       }, "线程 B").start();
   }
}

class DemoClass{
 //加减对象
   private int number = 0;
 /**
 * 加 1
 */
   public synchronized void increment() {
      try {
           while (number != 0){
           this.wait();
       }
           number++; System.out.println("--------" +Thread.currentThread().getName() + "加一成功----------,值为:" + number);
           notifyAll();
       }catch (Exception e){
           e.printStackTrace();
       }
    }
 /**
 * 减一
 */
   public synchronized void decrement(){
       try {
         while (number == 0){
         this.wait();
        }
          number--;
          System.out.println("--------" + Thread.currentThread().getName() + "减一成功----------,值为:" + number);
          notifyAll();
        }catch (Exception e){
             e.printStackTrace();
        }
    }
}

3.2 Lock 方案


import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
class DemoClass{
   //加减对象
   private int number = 0;
   //声明锁
   private Lock lock = new ReentrantLock();
   //声明钥匙
   private Condition condition = lock.newCondition();
   /**
   * 加 1
   */
    public void increment() {
        try {
          lock.lock();
          while (number != 0){
          condition.await();
         }
           number++; 
           System.out.println("--------" + Thread.currentThread().getName() + "加一成功----------,值为:" + number);
           condition.signalAll();
          }catch (Exception e){
             e.printStackTrace();
          }finally {
             lock.unlock();
          }
   }
   /**
   * 减一
   */
   public void decrement(){
      try {
        lock.lock();
        while (number == 0){
        condition.await();
       }
        number--;
        System.out.println("--------" + Thread.currentThread().getName() + "减一成功----------,值为:" + number);
        condition.signalAll();
       }catch (Exception e){
          e.printStackTrace();
       }finally {
         lock.unlock(); }
       }
 }

3.4 线程间定制化通信

面试

问题: A 线程打印 5 次 A,B 线程打印 10 次 B,C 线程打印 15 次 C,按照此顺序循环 10 轮


import java.util.concurrent.locks.Condition;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
class DemoClass{
   //通信对象:0--打印 A 1---打印 B 2----打印 C
   private int number = 0;
   //声明锁
   private Lock lock = new ReentrantLock();
   //声明钥匙 A
   private Condition conditionA = lock.newCondition();
   //声明钥匙 B
   private Condition conditionB = lock.newCondition();
   //声明钥匙 C
   private Condition conditionC = lock.newCondition(); /**
   * A 打印 5 次
   */
   public void printA(int j){
      try {
       lock.lock();
       while (number != 0){
       conditionA.await();
      }
        System.out.println(Thread.currentThread().getName() + "输出 A,第" + j + "轮开始");
        //输出 5 次 A
           for (int i = 0; i < 5; i++) {
              System.out.println("A");
            }
       //开始打印 B
       number = 1;
       //唤醒 B
       conditionB.signal();
       }catch (Exception e){
           e.printStackTrace();
       }finally {
         lock.unlock();
       }
   }
    /**
     * B 打印 10 次
    */
   public void printB(int j){
      try {
        lock.lock();
        while (number != 1){
        conditionB.await();
       }
         System.out.println(Thread.currentThread().getName() + "输出 B,第" + j + "轮开始");
        //输出 10 次 B
           for (int i = 0; i < 10; i++) {
               System.out.println("B");
            }
        //开始打印 C
         number = 2;
        //唤醒 C
        conditionC.signal();
       }catch (Exception e){
          e.printStackTrace();
       }finally {
         lock.unlock();
       }
 }
      /**
     * C 打印 15 次
     */
     public void printC(int j){
        try {
          lock.lock();
          while (number != 2){
          conditionC.await();
        }
          System.out.println(Thread.currentThread().getName() + "输出 C,第" + j + "轮开始");
        //输出 15 次 C
           for (int i = 0; i < 15; i++) {
                System.out.println("C");
            }
           System.out.println("-----------------------------------------");
           //开始打印 A
             number = 0;
           //唤醒 A
           conditionA.signal();
         }catch (Exception e){
            e.printStackTrace();
         }finally {
           lock.unlock();
         } 
     }
}

测试类


/**
* volatile 关键字实现线程交替加减
*/
public class TestVolatile {
 /**
 * 交替加减
 * @param args
 */
   public static void main(String[] args){
      DemoClass demoClass = new DemoClass();
      new Thread(() ->{
       for (int i = 1; i <= 10; i++) {
        demoClass.printA(i);
        }
      }, "A 线程").start();
      
       new Thread(() ->{
             for (int i = 1; i <= 10; i++) { 
                     demoClass.printB(i);
            }
      }, "B 线程").start();
      
       new Thread(() ->{
             for (int i = 1; i <= 10; i++) {
                     demoClass.printC(i);
              }
       }, "C 线程").start();
   }
}

在这里插入图片描述

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

真真最可爱

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值