Java多线程生产者消费者模式实现过程解析

 更新时间:2020年03月19日 14:22:31   投稿:yaominghui  
这篇文章主要介绍了Java多线程生产者消费者模式实现过程解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下

单生产者与单消费者

示例:

public class ProduceConsume {
    public static void main(String[] args) {
      String lock = new String("");
      Produce produce = new Produce(lock);
      Consume consume = new Consume(lock);

      new Thread(() -> {
        while (true) {
          produce.setValue();
        }
      }, "ProductThread").start();
      new Thread(() -> {
        while (true) {
          consume.getValue();
        }
      }, "ConsumeThread").start();
    }

    /**
     * 生产者
     */
    static class Produce {
      private String lock;

      public Produce(String lock) {
        this.lock = lock;
      }

      public void setValue() {
        try {
          synchronized (lock) {
            if (!ValueObject.value.equals("")) {
              lock.wait();
            }
            String value = System.currentTimeMillis() + "_" + System.nanoTime();
            System.out.println("set的值是" + value);
            ValueObject.value = value;
            lock.notify();
          }
        } catch (InterruptedException e) {
          e.printStackTrace();
        }
      }
    }

    /**
     * 消费者
     */
    static class Consume {
      private String lock;

      public Consume(String lock) {
        this.lock = lock;
      }

      public void getValue() {
        try {
          synchronized (lock) {
            if (ValueObject.value.equals("")) {
              lock.wait();
            }
            System.out.println("get的值是" + ValueObject.value);
            ValueObject.value = "";
            lock.notify();
          }
        } catch (InterruptedException e) {
          e.printStackTrace();
        }
      }
    }

    static class ValueObject {
      public static String value = "";
    }
}

执行结果如下:

多生产者与多消费者

这种模式下,容易出现“假死”,也就是全部线程都进入了 WAITNG 状态,程序不在执行任何业务功能了,整个项目呈停止状态。

示例:

public class MultiProduceConsume {
  public static void main(String[] args) throws InterruptedException {
    String lock = new String("");
    Produce produce = new Produce(lock);
    Consume consume = new Consume(lock);
    Thread[] pThread = new Thread[2];
    Thread[] cThread = new Thread[2];
    for (int i = 0; i < 2; i++) {
      pThread[i] = new Thread(() -> {
        while (true) {
          produce.setValue();
        }
      }, "生产者" + (i + 1));

      cThread[i] = new Thread(() -> {
        while (true) {
          consume.getValue();
        }
      }, "消费者" + (i + 1));
      pThread[i].start();
      cThread[i].start();
    }

    Thread.sleep(5000);
    Thread[] threadArray = new Thread[Thread.currentThread().getThreadGroup().activeCount()];
    Thread.currentThread().getThreadGroup().enumerate(threadArray);
    for (int i = 0; i < threadArray.length; i++) {
      System.out.println(threadArray[i].getName() + " " + threadArray[i].getState());
    }
  }

  static class Produce {
    private String lock;

    public Produce(String lock) {
      this.lock = lock;
    }

    public void setValue() {
      try {
        synchronized (lock) {
          while(!ValueObject.value.equals("")) {
            System.out.println("生产者 " + Thread.currentThread().getName() + " WAITING了⭐");
            lock.wait();
          }
          System.out.println("生产者 " + Thread.currentThread().getName() + " RUNNABLE了");
          String value = System.currentTimeMillis() + "_" + System.nanoTime();
          ValueObject.value = value;
          lock.notify();
        }
      } catch (InterruptedException e) {
        e.printStackTrace();
      }
    }
  }

  static class Consume {
    private String lock;

    public Consume(String lock) {
      this.lock = lock;
    }

    public void getValue() {
      try {
        synchronized (lock) {
          while (ValueObject.value.equals("")) {
            System.out.println("消费者 " + Thread.currentThread().getName() + " WAITING了⭐");
            lock.wait();
          }
          System.out.println("消费者 " + Thread.currentThread().getName() + "RUNNABLE了");
          ValueObject.value = "";
          lock.notify();
        }
      } catch (InterruptedException e) {
        e.printStackTrace();
      }
    }
  }

  static class ValueObject {
    public static String value = "";
  }
}

运行结果如图:

分析:

虽然代码中通过 wait/notify 进行通信了,但是不能保证 notify 唤醒的一定是异类,也可能是同类,比如“生产者”唤醒了“生产者”这样的情况。

解决方案:

假死出现的主要原因是有可能连续唤醒了同类。所以解决方案很简单,就是把 notify() 改为 notifyAll() 即可。

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

相关文章

  • Java反射机制的实现详解

    Java反射机制的实现详解

    反射主要解决动态编程,即使用反射时,所有的对象生成是动态的,因此调用的方法也是动态的.反射可以简化开发,但是代码的可读性很低
    2013-05-05
  • 浅谈hibernate中多表映射关系配置

    浅谈hibernate中多表映射关系配置

    下面小编就为大家带来一篇浅谈hibernate中多表映射关系配置。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-06-06
  • JavaMail与Spring整合过程解析

    JavaMail与Spring整合过程解析

    这篇文章主要介绍了JavaMail与Spring整合过程解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2019-11-11
  • SpringBoot集成thymeleaf渲染html模板的步骤详解

    SpringBoot集成thymeleaf渲染html模板的步骤详解

    这篇文章主要给大家详细介绍了SpringBoot集成thymeleaf如何使实现html模板的渲染,文中有详细的代码示例,具有一定的参考价值,需要的朋友可以参考下
    2023-06-06
  • 通过实例学习Either 树和模式匹配

    通过实例学习Either 树和模式匹配

    这篇文章主要介绍了通过实例学习Either 树和模式匹配,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,,需要的朋友可以参考下
    2019-06-06
  • 详解MyBatis的Dao层实现和配置文件深入

    详解MyBatis的Dao层实现和配置文件深入

    这篇文章主要为大家详细介绍了MyBatis的Dao层实现和配置文件深入,文中的示例代码讲解详细,感兴趣的小伙伴快来跟随小编一起学习一下
    2022-07-07
  • Spring中利用IOC实现注入的方式

    Spring中利用IOC实现注入的方式

    Spring IOC(控制反转)实现依赖注入,将对象创建和依赖关系的管理交由Spring容器处理,通过注解或XML配置,实现对象之间的松耦合,提高代码复用性和可维护性
    2023-04-04
  • 浅析java 循序与二元搜索算法

    浅析java 循序与二元搜索算法

    这篇文章主要简单介绍了java 循序与二元搜索算法,需要的朋友可以参考下
    2015-02-02
  • springboot中PostMapping正常接收json参数后返回404问题

    springboot中PostMapping正常接收json参数后返回404问题

    这篇文章主要介绍了springboot中PostMapping正常接收json参数后返回404问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-05-05
  • java sql ResultSet 之getRow()用法说明

    java sql ResultSet 之getRow()用法说明

    这篇文章主要介绍了java sql ResultSet 之getRow()用法说明,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-08-08

最新评论