生产者-消费者 BlockingQueue 运用示例

简单说明:

1、生产者负责将字符串转换成int 数字放入BlockingQueue,失败就停止生产消费线程。

2、消费者从BlockingQueue获得数字,取平方根值,并累积值。如果有负数,失败!停止生产消费线程。

3、模拟一个生产者,2个消费者。为了能均衡对应,生产者每次暂停 10毫秒,消费每次暂停23 毫秒~~

代码:

public class ThreadTest {

	private final BlockingQueue<Integer> queue = new ArrayBlockingQueue<Integer>( 5 );
	
	/**计算结果**/
	private final AtomicInteger atoInt = new AtomicInteger(0);
	
	/**整个计算过程是否成功**/
	private final AtomicBoolean isSuccess = new AtomicBoolean(true);
	
	private Thread t1 = null;
	
	private Thread t2 = null;
	
	private Thread t3 = null;
	
	private final CountDownLatch endGate = new CountDownLatch( 2 );

	public static void main(String[] args) throws InterruptedException {
		String[] data = new String[]{ "1", "1", "4", "9", "25", "36", "49", "64", "81", "144"};
		ThreadTest test = new ThreadTest();
		Producter p = test.new Producter(data);
		Consumer c1 = test.new Consumer();
		Consumer c2 = test.new Consumer();
		
		test.t1 = new Thread(p);
		test.t2 = new Thread(c1);
		test.t3 = new Thread(c2);
		
		test.start();
		test.endGate.await();
		if( test.isSuccess.get() ){
			System.out.println( "计算结果:" + test.atoInt.get() );
		}
		else{
			System.out.println( "计算过程中遇到错误!" );
		}
		
	}

	private void start(){
		t1.start();
		t2.start();
		t3.start();
	}
	
	private void stop(){
		t1.interrupt();
		t2.interrupt();
		t3.interrupt();
	}
	
	private class Producter implements Runnable{
		
		private final String[] data;
		
		public Producter(String[] data) {
			super();
			this.data = data;
		}

		@Override
		public void run() {
			try {
				for( String s : data ){
					//System.out.println( "当前读取数:" + s );
					int i = Integer.valueOf( s );
					
					queue.put( i );
					Thread.sleep( 10 );
				}
				
				queue.put( Integer.MAX_VALUE );
			} catch (NumberFormatException e) {
				e.printStackTrace();
				isSuccess.set(false);
				stop();
				return;
			} catch (InterruptedException e) {
				Thread.currentThread().interrupt();
			}
			
		}
		
		
	}
	
	
	private class Consumer implements Runnable{

		@Override
		public void run() {
			try {
				while( !Thread.currentThread().isInterrupted() ){
					int i = queue.take();
					Thread.sleep( 23 );
					//遇到错误值
					if( i<0 ){
						stop();
						isSuccess.set(false);
						break;
					}
					//结束标志
					else if( i == Integer.MAX_VALUE ){
						stop();
						break;
					}
					//System.out.println( Thread.currentThread() + "当前queue获取数:" + i );
					atoInt.addAndGet( (int)Math.sqrt(i) );
					
					//System.out.println( atoInt.get() );
					
				}
			} catch (InterruptedException e) {
				Thread.currentThread().interrupt();
			}
			finally{
				endGate.countDown();
			}
		}
		
	}
}


 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在Java中,"生产者-消费者"(Producer-Consumer)模式是一种经典的设计模式,用于解决多线程环境中的同步问题。它定义了两个角色,即生产者(Producer)和消费者(Consumer),它们分别负责创建数据(产品)和消费数据。 **1. 生产者(Producer):** 生产者通常在自己的线程中不断产生新的对象,并将这些对象添加到一个共享的队列(如`BlockingQueue`、`ArrayBlockingQueue`等)中。生产者需要处理可能的数据溢出(队列已满)和资源竞争(队列为空)。 **2. 消费者(Consumer):** 消费者从队列中取出数据并进行处理,消耗掉生产者产生的对象。同样,消费者需要处理可能的数据饥饿(队列空)和资源竞争(队列满)。 **3. 同步机制:** 为了确保生产者消费者之间的同步,Java提供了`synchronized`关键字或`Semaphore`、`CountDownLatch`、`CyclicBarrier`等并发工具。使用这些工具可以控制访问队列的权限,避免数据不一致或死锁。 **4. 示例代码:** 下面是一个简单的`BlockingQueue`实现的生产者消费者示例: ```java import java.util.concurrent.*; public class ProducerConsumer { private final BlockingQueue<String> queue = new LinkedBlockingQueue<>(10); private final Thread producer; private final Thread consumer; public ProducerConsumer() { producer = new Thread(() -> produceData()); consumer = new Thread(() -> consumeData()); producer.start(); consumer.start(); } private void produceData() { while (true) { synchronized (queue) { if (queue.isEmpty()) { try { queue.wait(); // 队列空时等待 } catch (InterruptedException e) { e.printStackTrace(); } } String data = generateData(); // 生产数据 queue.offer(data); // 添加到队列 System.out.println("Produced: " + data); queue.notify(); // 唤醒消费者 } } } private void consumeData() { while (true) { synchronized (queue) { if (queue.isEmpty()) { try { queue.wait(); // 队列空时等待 } catch (InterruptedException e) { e.printStackTrace(); } } String data = queue.take(); // 从队列获取数据 System.out.println("Consumed: " + data); } } } private String generateData() { // 实现数据生成逻辑 return "Data " + UUID.randomUUID().toString(); } public static void main(String[] args) { new ProducerConsumer(); } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值