java多线程_生产者消费者设计模式

概述:

(java的线程调度模式采用的是抢占式调度

生产者消费者是一个十分经典的多线程协作模式,弄懂生产者消费者问题能够让我们对多线程编程的理解更加深刻。

所谓生产者问题,实际上主要是包含了两类线程:

  • 一类是生产者线程用于生产数据;
  • 一类是消费者用于消费数据;
  • 为了解耦生产者和消费者的关系,通常会采用共享的**数据区域**,就像是一个仓库,生产者生产数据之后直接放置在共享区域中,并不需要关心消费者的行为。
  • 消费者只需要从共享区域中获取数据,并不需要关心生产者的行为。

在这里插入图片描述

为了体现生产和消费过程中的等待和唤醒,java中提供了几个方法供我们使用,这几个方法在Object类中,分别是wait()、notify()、notifyAll()三个方法。

三个核心点:

  • 任何对象中都一定有着三个方法(因为任意类都继承自Object类)
  • 只有作为锁对象的时候,才可以调用
  • 只有在同步的代码块中,才可以调用

其他情况下,调用一个对象的这三个方法,都会报错

  • void wait() :导致当前线程进入无限期等待状态,直到另一个线程调用该对象的notify()方法或notifyAll()方法。
  • void notify() : 随机唤醒一个正在等待对象监视器的线程。
  • void notifyAll() :唤醒正在等待对象监视器的所有线程。

需求:

奶箱类(Box),定义一个成员变量,表示第x瓶奶,提供存储牛奶和获取牛奶的操作。

生产者类(product),实现Runnable接口,重写run()方法,调用存储牛奶的操作。

消费者类(Customer),实现Runnable接口,重写run()方法,调用获取牛奶的操作。

测试类(Test),里面有main方法,main方法中的调用步骤如下:

  1. 创建奶箱对象,这是共享数据区域
  2. 创建生产者对象,把奶箱对象通过构造方法进行参数传递,因为在这个类中,要调用存储牛奶的操作。
  3. 创建消费者对象,把奶箱对象通过构造方法进行参数传递,因为在这个类中,要调用获取牛奶的操作
  4. 创建两个线程对象,分别把生产者和消费者对象作为构造方法参数传递。
  5. 启动线程

1、创建奶箱对象,这是共享数据区域

奶箱类中定义了一个奶箱数量milk和奶箱状态state,以及存储牛奶和获取牛奶的操作。

package CoreJava.day15_thread.product;

/**
 * Created by Intellij IDEA.
 *
 * @author zhudezhong
 * @date 2021/6/27 20:52
 */
//奶箱
public class Box {
    //定义牛奶的数量
    private int milk;
    //定义一个奶箱状态
    private boolean state = false;  //true表示奶箱有奶,false表示奶箱无奶

    //存取牛奶
    public synchronized void put(int milk) {
        if (state) {
            //奶箱有奶,则让停止往奶箱中放奶,等待消费者消费
            try {
                wait();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }

        //奶箱中没有奶,往其中放牛奶
        this.milk = milk;
        System.out.println("生产者生产了第" + milk + "瓶牛奶");

        //放完奶之后让奶箱的状态变为true
        state = true;

        //唤醒消费者,来拿牛奶
        notifyAll();
    }

    //拿到牛奶
    public synchronized void get() {
        //奶箱中没有牛奶,等待生产者生产牛奶
        if (!state){
            try {
                wait();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }

        //有牛奶,获取牛奶
        System.out.println("送奶工拿到了第" + milk + "瓶牛奶");
        //获取玩牛奶后改变奶箱状态
        state = false;
        //唤醒生产者来生产
        notifyAll();

    }

}

2、创建生产者对象,把奶箱对象通过构造方法进行参数传递,因为在这个类中,要调用存储牛奶的操作。

生产者类中,定义了一个有参构造方法,该参数是奶箱(Box)对象。这个参数传递的方式保证了我们传入的奶箱是同一个奶箱,创造一个了数据共享区域

package CoreJava.day15_thread.product;

/**
 * Created by Intellij IDEA.
 *
 * @author zhudezhong
 * @date 2021/6/27 20:52
 */
//生产者
public class Product implements Runnable {
    private Box box;
    //该构造方法可以确保传入的box奶箱是同一个对象
    public Product(Box box) {
        this.box = box;
    }

    @Override
    public void run() {

        //放入牛奶
        for (int i = 1; i <= 5; i++) {
            box.put(i);
        }
    }
}

3、创建消费者对象,把奶箱对象通过构造方法进行参数传递,因为在这个类中,要调用获取牛奶的操作

package CoreJava.day15_thread.product;

/**
 * Created by Intellij IDEA.
 *
 * @author zhudezhong
 * @date 2021/6/27 20:52
 */
//消费者
public class Custermor implements Runnable{
    private Box box;
    public Custermor(Box box) {
        this.box = box;
    }

    @Override
    public void run() {
        //一直取牛奶
        while (true){
            box.get();
        }
    }
}

4、测试类(Test)

package CoreJava.day15_thread.product;

/**
 * Created by Intellij IDEA.
 *
 * @author zhudezhong
 * @date 2021/6/27 20:53
 */
public class Test {
    public static void main(String[] args) {
        //创建奶箱对象
        Box box = new Box();
        //创建生产者、消费者任务对象
        Product product = new Product(box);
        Custermor custermor = new Custermor(box);

        //创建并启动生产者消费者线程
        Thread t1 = new Thread(product);
        Thread t2 = new Thread(custermor);
        t1.start();
        t2.start();
    }
}

运行结果:

在这里插入图片描述

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

A_Zhong20

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

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

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

打赏作者

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

抵扣说明:

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

余额充值