Java多线程学习笔记02:https://blog.csdn.net/weixin_44211980/article/details/106126778
4. 生产者消费者模式
4.1 生产者消费者模式概述
生产者消费者模式是一个十分经典的多线程协作的模式,其主要包含了两类线程
- 一类是生产者线程用于生产数据
- 一类是消费者线程用于消费数据
为了解耦生产者和消费者的关系,通常会采用共享的数据区域,就像是一个仓库,生产者生产数据后直接放置在共享区域中,并不关心消费者的行为,消费者只需从共享数据汇总去获取数据,并不需要关心生产者的行为
为了体现生产和消费过程中的等待和唤醒,Java在Object类中提供了以下方法:
方法名 | 说明 |
void wait() | 导致当前线程等待,直到另一个线程调用该对象的唤醒方法 |
void notify() | 唤醒正在等待对象监视器的单个线程 |
void notifyAll() | 唤醒正在等待对象监视器的所有线程 |
4.2 案例引入
案例描述
送奶工要将10瓶牛奶送入奶箱中,用户从奶箱中取出牛奶
解决思路
- 创建奶箱对象,将其作为共享区域
- 创建送奶工对象作为生产者,将奶箱对象传递进其构造方法中,并在“送奶”函数中调用它
- 创建用户对象作为消费者,将奶箱对象传递进其构造方法中,并在“取奶”函数中调用它
- 创建两个线程对象分别作为生产者和消费者进行模拟
- 启动线程
具体代码
public class Box {
private int milk = 0;
private boolean state = false; //表示奶箱中是否有牛奶
public synchronized void put_milk() {
if(state) { //奶箱中有奶等会再送
try {
wait();
} catch (InterruptedException e) {
// TODO 自动生成的 catch 块
e.printStackTrace();
}
}
++milk;
System.out.println(Thread.currentThread().getName() + "将第" + milk + "放入奶箱");
state = true; //此时奶箱中有奶了,改变状态
notifyAll(); //唤醒其他线程
}
public synchronized void get_milk() {
if(!state) { //奶箱中没奶等会再拿
try {
wait();
} catch (InterruptedException e) {
// TODO 自动生成的 catch 块
e.printStackTrace();
}
}
System.out.println(Thread.currentThread().getName() + "将第" + milk + "拿出奶箱");
state = false;
notifyAll();
}
}
public class Producter implements Runnable {
Box box;
public Producter() {};
public Producter(Box obj) {
box = obj;
}
@Override
public void run() {
for(int i = 0; i < 10; ++i) {
box.put_milk();
}
}
}
public class Consumer implements Runnable {
Box box;
public Consumer() {}
public Consumer(Box obj) {
box = obj;
}
@Override
public void run() {
while(true) {
box.get_milk();
}
}
}
public class ThreadStudy {
public static void Milk() {
Box b = new Box();
Producter p = new Producter(b);
Consumer c = new Consumer(b);
Thread t1 = new Thread(p, "A");
Thread t2 = new Thread(c, "B");
t1.start();
t2.start();
}
public static void main(String[] args) {
Milk();
}
}
运行结果
A将第1放入奶箱
B将第1拿出奶箱
A将第2放入奶箱
B将第2拿出奶箱
A将第3放入奶箱
B将第3拿出奶箱
A将第4放入奶箱
B将第4拿出奶箱
A将第5放入奶箱
B将第5拿出奶箱
A将第6放入奶箱
B将第6拿出奶箱
A将第7放入奶箱
B将第7拿出奶箱
A将第8放入奶箱
B将第8拿出奶箱
A将第9放入奶箱
B将第9拿出奶箱
A将第10放入奶箱
B将第10拿出奶箱