前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Java面试——写一个生产者与消费者

Java面试——写一个生产者与消费者

作者头像
Java架构师必看
发布2021-05-14 16:23:12
1.5K0
发布2021-05-14 16:23:12
举报
文章被收录于专栏:Java架构师必看Java架构师必看

一、通过synchronize 中的 wait 和 notify 实现


【1】我们可以将生产者和消费者需要的方法写在公共类中

代码语言:javascript
复制
package com.yintong.concurrent;

import java.util.LinkedList;

public class Concurrentcomm {
    //常量
    private static int MAX_VALUE = 10;
    //可以理解为缓存
    LinkedList<String> linkedList = new LinkedList<>();
    Object object = new Object();
    /*
     * 生产者方法
     */
    public void product() throws Exception {
        synchronized(linkedList) {
            while(MAX_VALUE == linkedList.size()) {
                System.out.println("仓库已满,【生产者】: 暂时不能执行生产任务!");
                linkedList.wait();
            }
            linkedList.push("   李四    ");
            System.out.println("【生产者】:生产了一个产品\t【现仓储量为】:" + linkedList.size());
            linkedList.notifyAll();
        }
    }
    /*
     * 消费者方法
     */
    public void customer() throws Exception {
        /*
         * 根据jdk的void notifyAll()的描述,“解除那些在该对象上调用wait()方法的线程的阻塞状态。该方法只能在同步方法或同步块内部调用。
         * 如果当前线程不是对象所得持有者,
         * 该方法抛出一个java.lang.IllegalMonitorStateException 异常”
         * so我们使用同一把锁
         */
        synchronized (linkedList) {
                        //多线程判断中使用 while 不要使用 if 否则会出现虚假唤醒问题
            while(linkedList.size() == 0) {
                System.out.println("仓库无货,【消费者】: 暂时不能执行消费任务!");
                linkedList.wait();
            }
            linkedList.pop();
            System.out.println("【消费者】:消费了一个产品\t【现仓储量为】:" + linkedList.size());
            linkedList.notifyAll();
        }
    }
}

【2】在 main 函数中调用生产者和消费者方法,并加限制即可

代码语言:javascript
复制
/**
 * @author zzx
 * @desc 生产者与消费者
 *
 */
public class Concurrent {
    //常量
    private static int MAX_VALUE = 100;

    public static void main(String[] args) {
        Concurrentcomm con = new Concurrentcomm();
        new Thread(new Runnable() {

            @Override
            public void run() {
                try {
                    for (int i = 0; i < MAX_VALUE; i++) {
                        Thread.sleep(0);
                        con.product();
                    }
                } catch (Exception e) {
                    // TODO Auto-generated catch block
                    e.printStackTrace();
                }
            }
        }).start();
        // 消费者
        new Thread(new Runnable() {

            @Override
            public void run() {
                try {
                    Thread.sleep(10);
                    for (int i = 0; i < MAX_VALUE; i++) {
                        con.customer();
                    }
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        }).start();
    }
}

【3】简单的生产者与消费者模式就完成了,可以看下运行的结果

二、通过 Lock 中的 await 与 signalAll 实现


【1】我们将公共的属性和方法放在 Resouce 类中,在资源类中使用 Lock 中的 lock()进行加锁,控制并发操作。使用 await()方法阻塞线程。使用 signalAll()唤醒线程。

代码语言:javascript
复制
/**
 * 通过 Lock 实现生产者与消费者
 * 资源类:将公共的资源放在一个单独的类中,可以将其看做一个产品,自身就就有生产和消费的能力(方法)
 */
public class ProductAndConsumer {
    public static void main(String[] args) {
        Resouce resouce = new Resouce();
        //生产者
        new Thread(()->{
            for (int i=1;i<=5;i++) {
                resouce.product();
            }
        },String.valueOf("生产者")) .start();

        //消费者
        new Thread(()->{
            for (int i=1;i<=5;i++){
                resouce.consumer();
            }
        },String.valueOf("消费者")).start();
    }
}
//资源类
class Resouce {
    private int MAX_VALUE = 3;
    private int MIN_VALUE = 0;
    private int number = 0;
    private Lock lock = new ReentrantLock();
    private Condition condition = lock.newCondition();

    //生产者
    public void product(){
        try {
            lock.lock();
            //如果生产的数量大于最大值则阻塞
            while(number >= MAX_VALUE){
                condition.await();
            }
            number++;
            System.out.println("【生产者】:生产了一个产品\t【现仓储量为】:" + number);
            condition.signalAll();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }finally {
            lock.unlock();
        }
    }

    //消费者
    public void  consumer(){
        try {
            lock.lock();
            //如果消费的值=0则阻塞
            while(number <= MIN_VALUE){
                condition.await();
            }
            number--;
            System.out.println("【消费者】:消费了一个产品\t【现仓储量为】:" + number);
            condition.signalAll();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }finally {
            lock.unlock();
        }
    }
}

【2】输出结果展示:

三、synchronized 和 Lock 的区别


【1】原始构成:synchronized 是关键字属于 JVM 层面。底层通过 monitorenter(进入)monitorexit(退出)实现。底层是通过 monitor 对象完成,其实 wait/notify 等方法也依赖于 monitor 对象,只有在同步块或方法中才能调用 wait/notify 等方法。Lock 是具体类(java.util.concurrent.locks.Lock)是 API 层面的锁。 【2】使用方法:synchronized 不需要用户手动释放锁,当 synchronized 代码执行完后,系统会自动释放锁。ReentrantLock 则需要用户手动释放锁,若未主动释放锁,就可能导致出现死锁的现象。 【3】等待是否中断:synchronized 不可中断,除非抛出异常或者正常运行完成。ReentrantLock 可中断,1)、设置超时时间 tryLock(long timeout,TimeUnit unit) 2)、lockInterruptibly() 放在代码块中,调用 interrupt() 方法可中断。 【4】加锁是否公平:synchronized 非公平锁。ReentrantLock 两者都可以,默认是非公平锁,构造方法可以传入 boolean 值,true 为公平锁,false 为非公平锁。 【5】锁绑定多个条件 Condition:synchronized 没有。ReentrantLock 用来实现分组唤醒需要唤醒的线程们,可以精确唤醒,而不是像 synchronized 要么随机唤醒一个线程要么唤醒全部线程。

四、通过阻塞队列实现生产者与消费者


【1】通过blockQueue 中的 put/take 方法实现生产者与消费者,具体实现如下:当生产者使用put 生产到指定的队列大小3时,就会阻塞当前线程。这是消费者线程会通过 take 方法消费队列中的消息。当队列中没有消息时,会阻塞,直到有消息消费。

代码语言:javascript
复制
public class BlockProductConsumer {
    public static void main(String[] args) {
        MyResouce resouce = new MyResouce(new ArrayBlockingQueue(3));
        //生产者线程
        new Thread(()->{
            for(int i=1;i<=10;i++){
                resouce.product();
            }
        },"生产者").start();

        //消费者线程
        new Thread(()->{
            for(int i=1;i<=10;i++){
                try {
                    resouce.consumer();
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        },"消费者").start();

        try {
            TimeUnit.SECONDS.sleep(1);
            resouce.stop();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}


/**
 * 公共资源类
 */
class MyResouce{
    //标记 while 无限循环
    private volatile boolean FLAG = true;
    //队列中存入的数值
    private AtomicInteger atomicInteger = new AtomicInteger();
    //组合一个阻塞队列,通过构造器传入
    private BlockingQueue blockingQueue;
    public MyResouce(BlockingQueue blockingQueue) {
        this.blockingQueue = blockingQueue;
    }

    //生产者
    public void product(){
        try {
            while (FLAG){
                blockingQueue.put(String.valueOf(atomicInteger.incrementAndGet()));
                System.out.println("生产者生产第"+blockingQueue.size()+"个产品");
            }
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }

    //消费者
    public void consumer() throws InterruptedException {
        while (FLAG){
            blockingQueue.take();
            System.out.println("消费者消费第"+(blockingQueue.size()+1)+"个产品");
        }
    }

    public void stop(){
        FLAG = false;
        System.out.println("========================");
    }
}

【2】效果展示:

​本文来源程序猿进阶,由javajgs_com转载发布,观点不代表Java架构师必看的立场,转载请标明来源出处

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 一、通过synchronize 中的 wait 和 notify 实现
  • 二、通过 Lock 中的 await 与 signalAll 实现
  • 三、synchronized 和 Lock 的区别
  • 四、通过阻塞队列实现生产者与消费者
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档