`
kingj
  • 浏览: 426449 次
  • 性别: Icon_minigender_1
  • 来自: 成都
社区版块
存档分类
最新评论

java线程生产者与消费者实例(使用List实现同步)

 
阅读更多

学习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生产者消费者模式实例分析

    在Java中,可以使用Runnable接口来实现消费者线程。在上面的代码中,Consumer类就是一个消费者,它实现了Runnable接口,负责从缓存中获取产品并进行消费。 四、缓存 缓存是一个负责存储产品的对象。在Java中,可以...

    java生产者-消费者

    通过上述分析可以看出,Java中的生产者-消费者模式是一种非常实用且高效的多线程编程模式,它能够帮助开发者更好地管理线程间的同步问题。本示例中的代码实现了基本的功能,并展示了如何使用 `synchronized`、`wait...

    Nachos的线程管理模块升级

    为了验证线程管理模块的正确性,可以编写实现经典同步互斥问题的实例程序,如"哲学家就餐"问题。在这种问题中,五个哲学家线程共享五根筷子,需要遵循特定的规则避免死锁。通过使用信号量机制,可以确保哲学家们正确...

    Java高级编程实例 100个

    实例可能包括模拟银行转账、生产者消费者模型等。 2. **网络编程**:Java提供了Socket编程接口,可以创建TCP和UDP连接。通过实例学习如何实现服务器和客户端的通信,如搭建简单的HTTP服务器或文件传输应用。 3. **...

    java源码包---java 源码 大量 实例

     Java实现的FTP连接与数据浏览程序,实现实例化可操作的窗口。  部分源代码摘录:  ftpClient = new FtpClient(); //实例化FtpClient对象  String serverAddr=jtfServer.getText(); //得到服务器地址  ...

    java多线程设计模式详解

    3. 生产者-消费者(Producer-Consumer)模式:这是一种典型的线程协作模式,其中生产者负责生产数据并放入缓冲区,消费者从缓冲区取出数据消费。该模式适用于数据处理流水线,以及需要异步处理的场景。 二、线程...

    java编程基础,应用与实例

    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接口 ...

    Java开发技术大全(500个源代码).

    producer_consumer.java 演示生产者-消费者线程 consumer.java 消费者线程 producer.java 生产者线程 common.java 公有类 第9章 示例描述:本章学习运行时类型识别。 Candy.java 一个用来测试的简单类 ...

    java学习(基于Java阻塞队列的搜索实例).pdf

    阻塞队列常用于生产者-消费者场景中,保证了线程间的协调。 本实例中使用了`ArrayBlockingQueue`,这是最常用的阻塞队列实现之一,它是一个有界队列,能够按FIFO(先进先出)的顺序处理元素。创建`...

    Lock、Condition实现简单的生产者消费者模式示例

    在Java多线程编程中,生产者消费者模式是一种常见的设计模式,用于协调生产者线程和消费者线程之间的协作,确保资源的有效利用和同步。在这个示例中,使用了Java的`java.util.concurrent.locks`包中的`Lock`接口和`...

    java 之十大经典案例

    使用`wait()`和`notify()`方法,可以实现生产者线程与消费者线程之间的协同工作,使得数据的生产和消费有序进行,避免资源浪费。 3. **异常处理**: Java提供了丰富的异常处理机制,包括`try-catch-finally`块和`...

    Java JDK实例宝典

    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协议发送...

    java初学者必看

    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 异常捕获...

    Java程序设计实验与实训源代码

    实训项目可能包含并发编程的案例,如实现生产者消费者模型或线程间的协作。 7. **图形用户界面(GUI)**: "java课程设计实例.exe"可能包含使用Java Swing或JavaFX库创建的GUI应用程序,让学生了解如何构建交互式用户...

    java源码包3

     Java实现的FTP连接与数据浏览程序,实现实例化可操作的窗口。  部分源代码摘录:  ftpClient = new FtpClient(); //实例化FtpClient对象  String serverAddr=jtfServer.getText(); //得到服务器...

    java源码包2

     Java实现的FTP连接与数据浏览程序,实现实例化可操作的窗口。  部分源代码摘录:  ftpClient = new FtpClient(); //实例化FtpClient对象  String serverAddr=jtfServer.getText(); //得到服务器...

    JAVA经典教材笔记

    - **线程操作案例—生产者和消费者** - 生产者-消费者模型:使用阻塞队列实现。 - 生产者-消费者模型的优点:解耦生产者与消费者。 - **线程生命周期** - 线程的状态:NEW、RUNNABLE、BLOCKED等。 - 线程状态...

    Java通过wait()和notifyAll()方法实现线程间通信

    在提供的代码示例中,`ProdCons`类使用了一个`LinkedList`作为共享资源,用于生产者(`Producer`)和消费者(`Consumer`)线程之间的通信。`LinkedList`是一个线程不安全的数据结构,但在这个例子中,通过在访问`list`...

    Java面试题集附答案

    同时,使用`wait()`和`notifyAll()`方法来协调生产者和消费者之间的同步问题。 ### 循环赛问题实现 示例代码还给出了一个循环赛问题的部分实现。该问题的目标是模拟一个从1到`total`的数字链表,按照给定步长`k`...

Global site tag (gtag.js) - Google Analytics