Java實現(xiàn)簡易生產(chǎn)者消費(fèi)者模型過程解析
一、概述
一共兩個線程,一個線程生產(chǎn)產(chǎn)品,一個線程消費(fèi)產(chǎn)品,使用同步代碼塊方法,同步兩個線程。當(dāng)產(chǎn)品沒有時,通知生產(chǎn)者生產(chǎn),生產(chǎn)者生產(chǎn)后,通知消費(fèi)者消費(fèi),并等待消費(fèi)者消費(fèi)完。
需要注意的是,有可能出現(xiàn),停止生產(chǎn)產(chǎn)品后,消費(fèi)者還沒未來得及消費(fèi)生產(chǎn)者生產(chǎn)的最后一個產(chǎn)品,就結(jié)束消費(fèi),導(dǎo)致最后一個產(chǎn)品沒有被消費(fèi)。
本例使用synchronize以及wait()、notify()實現(xiàn)簡易版的線程者消費(fèi)者模型。
二、測試用例
這里的產(chǎn)品用筆來演示,每只筆都有其編號code
一共有四個類:分別是生產(chǎn)者類,產(chǎn)品類,消費(fèi)者類,測試類

產(chǎn)品
package test.exception.producer_consumer_model;
/*
假設(shè)為產(chǎn)品為筆
*/
public class Production {
private String type = "";
private String color = "";
private long code = 0; // 產(chǎn)品編號
private boolean isProduced = false; // 是否生產(chǎn)完成 初始狀態(tài)為未生產(chǎn)狀態(tài)
private boolean isContinueProduce = true; // 是否停產(chǎn)該產(chǎn)品
public void setContinueProduce(boolean continueProduce) {
isContinueProduce = continueProduce;
}
public void setCode(long code) {
this.code = code;
}
public Production(){
}
public boolean isContinueProduce() {
return isContinueProduce;
}
public void setType(String type) {
this.type = type;
}
public void setColor(String color) {
this.color = color;
}
public void setProduced(boolean produced) {
isProduced = produced;
}
public boolean isProduced() {
return isProduced;
}
@Override
public String toString() {
return color + type + "-" + code;
}
}
生產(chǎn)者
package test.exception.producer_consumer_model;
public class Producer implements Runnable {
private final Production pen; // 產(chǎn)品
public Producer(Production pen) {
this.pen = pen;
}
// 生產(chǎn)
public void produce() {
long code = 0;
while (this.pen.isContinueProduce()) {
synchronized (this.pen) {
if (this.pen.isProduced()) {
try {
this.pen.wait(); // 等待消費(fèi)者消費(fèi)
} catch (InterruptedException e) {
e.printStackTrace();
}
}
// 開始生產(chǎn)
this.pen.setType("鉛筆");
this.pen.setColor("藍(lán)色");
this.pen.setCode(code++);
this.pen.setProduced(true);
System.out.println(this.pen + " is produced");
this.pen.notify();
}
}
System.out.println("finish producing");
}
@Override
public void run() {
produce();
}
}
消費(fèi)者
package test.exception.producer_consumer_model;
public class Consumer implements Runnable {
private final Production pen;
public Consumer(Production pen) {
this.pen = pen;
}
// 持續(xù)消費(fèi)
public void consumer() {
while (this.pen.isContinueProduce()) {
synchronized (this.pen) {
if (!this.pen.isProduced()) {
try {
this.pen.wait(); // 等待生產(chǎn)者生產(chǎn)
} catch (InterruptedException e) {
e.printStackTrace();
}
}
System.out.println(this.pen + " is consumed"); // 使用
this.pen.setProduced(false); // 使用完后更新狀態(tài)
this.pen.notify();
}
}
// 確保停止生產(chǎn)后,能夠使用最后生產(chǎn)的一支筆
if (this.pen.isProduced()) {
System.out.println(this.pen + " is consumed");
}
System.out.println("finish using");
}
@Override
public void run() {
consumer();
}
}
主線程測試
package test.exception.producer_consumer_model;
public class Demo {
public static void main(String[] args) throws InterruptedException {
Production pen = new Production();
Consumer consumer = new Consumer(pen);
Producer producer = new Producer(pen);
new Thread(producer).start(); // 開啟生產(chǎn)者線程
new Thread(consumer).start(); // 開啟消費(fèi)者線程
Thread.sleep(10000);
pen.setContinueProduce(false); // 10s后停止生產(chǎn)該類型的筆
}
}
運(yùn)行結(jié)果

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
spring boot使用thymeleaf跳轉(zhuǎn)頁面實例代碼
本篇文章主要介紹了spring boot使用thymeleaf跳轉(zhuǎn)頁面,實例介紹了thymeleaf的原理和介紹,有興趣的可以了解一下。2017-04-04
idea設(shè)置在包里面在創(chuàng)建一個包方式
這篇文章主要介紹了idea設(shè)置在包里面在創(chuàng)建一個包方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-05-05
深入學(xué)習(xí)java位運(yùn)算的基礎(chǔ)知識
位運(yùn)算是直接對整數(shù)在內(nèi)存中的二進(jìn)制位進(jìn)行操作嗎,位運(yùn)算即可以節(jié)約內(nèi)存,同時使程序速度更快效率更高。文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,下面我們來一起學(xué)習(xí)下吧2019-06-06

