public class Box {
// 定义一个变量 表示X瓶奶
private int milk;
// 表示一个变量,表示奶箱的状态
private boolean state = false;
// 送奶
public synchronized void put(int milk) throws InterruptedException {
// 有牛奶
if(state){
try {
wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
// 没有牛奶就添加牛奶
this.milk = milk;
System.out.println("送奶工将第" + this.milk + "瓶奶放入奶箱");
// 生产完毕后,修改奶的状态
this.state = true ;
// 唤醒其他等待的线程
notifyAll();
}
// 获取奶
public synchronized void get() throws InterruptedException {
// 如果没有奶,就等待生产
if(!state){
try {
wait();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
//如果有牛奶,就消费牛奶
System.out.println("用户拿到第" + this.milk + "瓶奶");
// 修改奶箱状态
this.state = false;
// 唤醒等待线程
notifyAll();
}
}
public class Cunstomer implements Runnable{
private Box b;
public Cunstomer(Box b) {
this.b = b;
}
/**
* 重写run方法
*/
@Override
public void run() {
try {
b.get();
} catch (InterruptedException e) {
e.printStackTrace();
}
//没有这个true 会执行不下去 为了让线程循环执行下去 这里也可使用 for for 和while均可
for(int i = 1 ; i<=30;i++) {
try {
b.get();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
public class Producer implements Runnable{
private Box b;
public Producer(com.thmeet.content.mythread.itheima_12.test.Box b) {
this.b = b;
}
@Override
public void run() {
for (int i = 1; i <= 30; i++){
try {
System.out.println("生产第"+i+ "瓶奶");
b.put(i);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
/*
生产者消费者案例中包含的类:
1:奶箱类(Box):定义一个成员变量,表示第x瓶奶,提供存储牛奶和获取牛奶的操作
2:生产者类(Producer):实现Runnable接口,重写run()方法,调用存储牛奶的操作
3:消费者类(Customer):实现Runnable接口,重写run()方法,调用获取牛奶的操作
4:测试类(BoxDemo):里面有main方法,main方法中的代码步骤如下
A:创建奶箱对象,这是共享数据区域
B:创建生产者对象,把奶箱对象作为构造方法参数传递,因为在这个类中要调用存储牛奶的操作
C:创建消费者对象,把奶箱对象作为构造方法参数传递,因为在这个类中要调用获取牛奶的操作
D:创建2个线程对象,分别把生产者对象和消费者对象作为构造方法参数传递
E:启动线程
*/
public class BoxDemo {
public static void main(String[] args) {
//创建奶箱对象,这是共享数据区域
Box b= new Box();
//创建生产者对象,把奶箱对象作为构造方法参数传递,因为在这个类中要调用存储牛奶的操作
Producer p = new Producer(b);
//创建消费者对象,把奶箱对象作为构造方法参数传递,因为在这个类中要调用获取牛奶的操作
Cunstomer c = new Cunstomer(b);
//创建2个线程对象,分别把生产者对象和消费者对象作为构造方法参数传递
Thread pt = new Thread(p);
Thread ct = new Thread(c);
//启动线程
ct.start();
pt.start();
}
}