学习java线程的时候,看到生产者与消费者例子,有感而发。
下面是模拟汽车生产商和顾客(权当消费者可以直接从厂家买到汽车),废话不说,上代码。
package com.zx.thread.work;
import java.io.FileNotFoundException;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
/**
* 我们模拟一个生产者和消费者购买汽车的例子
*
* @author maping
*
*/
public class ProducerAndConsumer {
//厂家存放汽车的车间
private static List<Car> cars = new ArrayList<Car>();
static void test(int producers, int consumers) {
ExecutorService exec = Executors.newCachedThreadPool();
for (int i = 0; i < producers; i++) {
// 上海大众的producers个分厂分别生产了100台polo车
exec.execute(new CarProducer("上海大众", cars, 100, i));
}
for (int i = 0; i < consumers; i++) {
// 消费者张三的consumers亲戚分别买10台polo
exec.execute(new CarConsumer("张三", cars, 10, i));
}
exec.shutdown();
}
public static void main(String[] args) throws FileNotFoundException {
test(100, 1000);
}
}
/**
汽车类
*/
class Car {
private String name = "polo";
private int id;
private CarProducer producer;
public Car() {
}
public Car(CarProducer producer, String name, int id) {
this.producer = producer;
this.name = name;
this.id = id;
}
public String toString() {
return this.producer.getName() + "#" + this.producer.getId() + "#"
+ this.getClass().getSimpleName() + "#" + name + "#" + id;
}
}
/*
汽车制造厂商
*/
class CarProducer implements Runnable {
private int count;
private List<Car> cars;
private String name;
private int id;
public CarProducer(String name, List<Car> cars, int count, int id) {
this.name = name;
this.cars = cars;
this.count = count;
this.id = id;
}
public void addCar(int ids) {
System.out.println(this.name + "#" + id + ">>开始生产汽车" + ids);
//此处同步cars,如果使用BlockingQueue,则无需同步代码
synchronized (cars) {
cars.add(new Car(this, name, ids));
//车间里面有了刚刚制造好的汽车,那么就可以通知消费者来买了
cars.notifyAll();
}
}
public void run() {
for (int i = 0; i < count; i++) {
this.addCar(i);
}
}
public String getName() {
return name;
}
public int getId() {
return id;
}
}
/**
消费者
*/
class CarConsumer implements Runnable {
private List<Car> cars;
private int count;
private String name;
private int id;
public CarConsumer(String name, List<Car> cars, int count, int id) {
this.name = name;
this.cars = cars;
this.count = count;
this.id = id;
}
public void buyCar() throws InterruptedException {
//消费者购买汽车时,锁定汽车对象
synchronized (cars) {
System.out.println(name + "#" + id + ">>" + "开始选车");
int waitTime = 10 * 1000;
//消费者刚刚来到汽车存放的车间时,如果发现
//车间还没有汽车,那么他只能等待,这里设置了超时时间,如果
//超过指定时间,则一气走之,相当于抛出异常了
while (cars.isEmpty()) {
System.out.println("[" + name + "#" + id + "]当前车缺货,等待"
+ waitTime + "毫秒");
wait(waitTime);
}
//有汽车来了,则个时候就可以买汽车了
Car car = cars.remove(0);
System.out.println("[" + name + "#" + id + "]" + "买到车了" + car);
System.out.println("***************************");
System.err.println("当前库存为\n" + cars.size());
}
}
public void run() {
while (count-- > 0) {
try {
buyCar();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
这段代码在生产者线程制造的汽车数>=消费者线程购买的汽车数时,没有任何问题,当反之的时候,就会由于
等待制造商生产汽车,而此时制造商或许不再制造汽车了,那么就会抛出异常。
我在思考,比如MQ的消息队列是怎么做到那么高效率呢
分享到:
相关推荐
在Java中,可以使用Runnable接口来实现消费者线程。在上面的代码中,Consumer类就是一个消费者,它实现了Runnable接口,负责从缓存中获取产品并进行消费。 四、缓存 缓存是一个负责存储产品的对象。在Java中,可以...
通过上述分析可以看出,Java中的生产者-消费者模式是一种非常实用且高效的多线程编程模式,它能够帮助开发者更好地管理线程间的同步问题。本示例中的代码实现了基本的功能,并展示了如何使用 `synchronized`、`wait...
为了验证线程管理模块的正确性,可以编写实现经典同步互斥问题的实例程序,如"哲学家就餐"问题。在这种问题中,五个哲学家线程共享五根筷子,需要遵循特定的规则避免死锁。通过使用信号量机制,可以确保哲学家们正确...
实例可能包括模拟银行转账、生产者消费者模型等。 2. **网络编程**:Java提供了Socket编程接口,可以创建TCP和UDP连接。通过实例学习如何实现服务器和客户端的通信,如搭建简单的HTTP服务器或文件传输应用。 3. **...
Java实现的FTP连接与数据浏览程序,实现实例化可操作的窗口。 部分源代码摘录: ftpClient = new FtpClient(); //实例化FtpClient对象 String serverAddr=jtfServer.getText(); //得到服务器地址 ...
3. 生产者-消费者(Producer-Consumer)模式:这是一种典型的线程协作模式,其中生产者负责生产数据并放入缓冲区,消费者从缓冲区取出数据消费。该模式适用于数据处理流水线,以及需要异步处理的场景。 二、线程...
14.7.3 生产者与消费者 221 14.7.4 多消费者 224 14.8 stop()、suspend()和resume() 225 14.9 巩固练习 226 第15章 常用API之二 228 15.1 Collection接口 228 15.1.1 Set接口 228 15.1.2 List接口 ...
producer_consumer.java 演示生产者-消费者线程 consumer.java 消费者线程 producer.java 生产者线程 common.java 公有类 第9章 示例描述:本章学习运行时类型识别。 Candy.java 一个用来测试的简单类 ...
阻塞队列常用于生产者-消费者场景中,保证了线程间的协调。 本实例中使用了`ArrayBlockingQueue`,这是最常用的阻塞队列实现之一,它是一个有界队列,能够按FIFO(先进先出)的顺序处理元素。创建`...
在Java多线程编程中,生产者消费者模式是一种常见的设计模式,用于协调生产者线程和消费者线程之间的协作,确保资源的有效利用和同步。在这个示例中,使用了Java的`java.util.concurrent.locks`包中的`Lock`接口和`...
Java实现的FTP连接与数据浏览程序 1个目标文件 摘要:Java源码,网络相关,FTP Java实现的FTP连接与数据浏览程序,实现实例化可操作的窗口。 部分源代码摘录: ftpClient = new FtpClient(); //实例化FtpClient对象 ...
使用`wait()`和`notify()`方法,可以实现生产者线程与消费者线程之间的协同工作,使得数据的生产和消费有序进行,避免资源浪费。 3. **异常处理**: Java提供了丰富的异常处理机制,包括`try-catch-finally`块和`...
19 线程——BlockingQueue 第17章 Java与XML 17. 1 用DOM处理XML文档 17. 2 用SAX处理XML文档 17. 3 用XSLT转换XML 17. 4 对象与XML的转换 第18章 Java Mail 18. 1 使用SMTP协议发送...
11.5.3 运行生产者/消费者 11.6 死锁 11.7 本章习题 第12章 异常处理 12.1 异常的概念 12.2 异常的基本样式 12.3 Java异常类 12.3.1 异常类层次结构 12.3.2 异常处理方法 12.4 异常捕获 12.4.1 异常捕获...
实训项目可能包含并发编程的案例,如实现生产者消费者模型或线程间的协作。 7. **图形用户界面(GUI)**: "java课程设计实例.exe"可能包含使用Java Swing或JavaFX库创建的GUI应用程序,让学生了解如何构建交互式用户...
Java实现的FTP连接与数据浏览程序,实现实例化可操作的窗口。 部分源代码摘录: ftpClient = new FtpClient(); //实例化FtpClient对象 String serverAddr=jtfServer.getText(); //得到服务器...
Java实现的FTP连接与数据浏览程序,实现实例化可操作的窗口。 部分源代码摘录: ftpClient = new FtpClient(); //实例化FtpClient对象 String serverAddr=jtfServer.getText(); //得到服务器...
- **线程操作案例—生产者和消费者** - 生产者-消费者模型:使用阻塞队列实现。 - 生产者-消费者模型的优点:解耦生产者与消费者。 - **线程生命周期** - 线程的状态:NEW、RUNNABLE、BLOCKED等。 - 线程状态...
在提供的代码示例中,`ProdCons`类使用了一个`LinkedList`作为共享资源,用于生产者(`Producer`)和消费者(`Consumer`)线程之间的通信。`LinkedList`是一个线程不安全的数据结构,但在这个例子中,通过在访问`list`...