`
endual
  • 浏览: 3560955 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

java多线程 阻塞队列的使用的 arrayBlockQueue

    博客分类:
  • java
 
阅读更多

我们在平时的时候使用的是队列,指定好个数以后,如果放的数据超过了队列设定的个数的时候会报错误的。

那么多线程中,有一个阻塞队列的叫arrayBlockQueue。这个类是对queue在多线程中使用的扩展,

也就是说,当作为临界资源的时候,这个队列是安全,存放数据如果超过了队列设定好的初始的数据的时候,

放入数据的线程将会被等待着的。

下面来看例子。

 

场景模拟:

现在有两个线程,一个线程往队列里面添加数据,一个是将队列里面的数据取出来并且删掉数据。

当然,放数据的线程要速度快点,这样才能够模拟出来队列里面存放的数据的个数会超过队列起先设定好的初始的个数吧

存放的线程是每个 5 秒钟放一个数据,取得话是每个10秒钟 取一次,下面是代码 三个类:

 

 

main类

 

package endual;

import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

/**
 * 阻塞对列,往对立里面添加元素
 * 一个由数组支持的有界阻塞队列。此队列按 FIFO(先进先出)
 * 原则对元素进行排序。队列的头部 是在队列中存在时间最长的元素。
 * 队列的尾部 是在队列中存在时间最短的元素。新元素插入到队列的尾部,
 * 队列检索操作则是从队列头部开始获得元素。 
 *
 *题目:
 *两个操作队列,一个队列是取数据,一个队列是放数据,如果一个线程取放数据过快,那么对立会阻塞,
 *线程也会进行等待着的
 *
 *
 */
public class ArrayBlockQueueApp {

	public static void main(String[] args) {
		ArrayBlockQueueApp app = new ArrayBlockQueueApp() ;
		app.mian() ;
	}
	
	
	private void mian() {

		ExecutorService es = Executors.newCachedThreadPool() ;
		ArrayBlockingQueue<String> abq = new ArrayBlockingQueue<String>(10) ;
		ThreadGet t1 = new ThreadGet(abq);
		Thread    t2 = new Thread(new ThreadPut(abq)) ;
		
		es.execute(t1) ;
		es.execute(t2) ;

		
		
	}
}

 

 

 

get线程类

 

package endual;

import java.util.concurrent.ArrayBlockingQueue;

public class ThreadGet extends Thread{

	ArrayBlockingQueue<String> abq = null ;
	public ThreadGet(ArrayBlockingQueue<String> abq) {
		this.abq = abq ;
	}
	
	public void run(){
		
		while (true) {
			
			try {
				Thread.sleep(10000) ;
				System.out.println("我要从队列中取数据了");
				String msg = abq.remove() ;
				System.out.println("队列里面取得的数据是:" + msg + " 队列中还的数据个数还有的 :" + abq.size());
				
			} catch (InterruptedException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			}//每个十秒钟会进行拿数据出来的
			
			
		}
		
		
	}
}

 

 

取数据的线程类:

 

package endual;

import java.util.concurrent.ArrayBlockingQueue;

public class ThreadPut implements Runnable{

	private ArrayBlockingQueue<String> abq = null ;
	public ThreadPut (ArrayBlockingQueue<String> abq) {
		this.abq = abq ;
	}
	public void run() {
		
		while (true) {
			
			try {
				System.out.println("我要向队列里面存放数据了");
				Thread.sleep(5000) ;
				abq.put("1") ;
				System.out.println("当前的队列里面存放的数据的个数是:" + abq.size());
			} catch (InterruptedException e) {
				// TODO Auto-generated catch block
				e.printStackTrace();
			} //向队列里面存放数据
			
			
			
			
		}
		
		
	}
	
	
	
}

 

 

下面是测试的结果:

 

我要向队列里面存放数据了
当前的队列里面存放的数据的个数是:1
我要向队列里面存放数据了
我要从队列中取数据了
队列里面取得的数据是:1 队列中还的数据个数还有的 :0
当前的队列里面存放的数据的个数是:1
我要向队列里面存放数据了
当前的队列里面存放的数据的个数是:2
我要向队列里面存放数据了
当前的队列里面存放的数据的个数是:3
我要向队列里面存放数据了
我要从队列中取数据了
队列里面取得的数据是:1 队列中还的数据个数还有的 :2
当前的队列里面存放的数据的个数是:3
我要向队列里面存放数据了
我要从队列中取数据了
队列里面取得的数据是:1 队列中还的数据个数还有的 :2
当前的队列里面存放的数据的个数是:3
我要向队列里面存放数据了
当前的队列里面存放的数据的个数是:4
我要向队列里面存放数据了
当前的队列里面存放的数据的个数是:5
我要向队列里面存放数据了
我要从队列中取数据了
队列里面取得的数据是:1 队列中还的数据个数还有的 :4
当前的队列里面存放的数据的个数是:5
我要向队列里面存放数据了
当前的队列里面存放的数据的个数是:6
我要向队列里面存放数据了
我要从队列中取数据了
队列里面取得的数据是:1 队列中还的数据个数还有的 :5
当前的队列里面存放的数据的个数是:6
我要向队列里面存放数据了
当前的队列里面存放的数据的个数是:7
我要向队列里面存放数据了
我要从队列中取数据了
队列里面取得的数据是:1 队列中还的数据个数还有的 :6
当前的队列里面存放的数据的个数是:7
我要向队列里面存放数据了
当前的队列里面存放的数据的个数是:8
我要向队列里面存放数据了
我要从队列中取数据了
队列里面取得的数据是:1 队列中还的数据个数还有的 :7
当前的队列里面存放的数据的个数是:8
我要向队列里面存放数据了
当前的队列里面存放的数据的个数是:9
我要向队列里面存放数据了
我要从队列中取数据了
队列里面取得的数据是:1 队列中还的数据个数还有的 :8
当前的队列里面存放的数据的个数是:9
我要向队列里面存放数据了
当前的队列里面存放的数据的个数是:10
我要向队列里面存放数据了
我要从队列中取数据了
队列里面取得的数据是:1 队列中还的数据个数还有的 :9
当前的队列里面存放的数据的个数是:10
我要向队列里面存放数据了
我要从队列中取数据了
当前的队列里面存放的数据的个数是:10
我要向队列里面存放数据了
队列里面取得的数据是:1 队列中还的数据个数还有的 :10
我要从队列中取数据了
当前的队列里面存放的数据的个数是:10
我要向队列里面存放数据了
队列里面取得的数据是:1 队列中还的数据个数还有的 :10
我要从队列中取数据了
当前的队列里面存放的数据的个数是:10
我要向队列里面存放数据了
队列里面取得的数据是:1 队列中还的数据个数还有的 :10
我要从队列中取数据了
当前的队列里面存放的数据的个数是:10
我要向队列里面存放数据了
队列里面取得的数据是:1 队列中还的数据个数还有的 :10
部分的结果。。。。。。

 

 

我们从结果中可以看出来,当队列里面的数据达到10个的时候,存放数据的线程在存放的时候将等待队列少的位子,不会像队列里面的数据少于10个的时候,存放数据是无需等待的,5秒钟到了就马上放进去。而到了10个数据的时候,存放的那个现在到了5秒钟的时候,还必须在等待位子,因为队列中还是10个位子,只有当取数据的线程取出数据以后,空出来的

位子后马上就添加进去了的

 

 

 

分享到:
评论
4 楼 FYIHDG 2013-12-25  
3 楼 endual 2012-07-06  
502220545 写道
502220545 写道
您好 看了您写的例子 有一点小小的建议
1 当放数据的睡眠时间 大于 取数据的睡眠时间时 就会跑出异常

可以将 arrayBlockingQueue的 add和remove方法 换成 put 和 take 这样就不会报异常了 是以阻塞的形式运行的

sorry 看错了 然后自己写错了 抱歉


没关系,我的文章很少有朋友给我提建议的,我大多是转的。呵呵。
这篇到是自己写的
2 楼 502220545 2012-07-06  
502220545 写道
您好 看了您写的例子 有一点小小的建议
1 当放数据的睡眠时间 大于 取数据的睡眠时间时 就会跑出异常

可以将 arrayBlockingQueue的 add和remove方法 换成 put 和 take 这样就不会报异常了 是以阻塞的形式运行的

sorry 看错了 然后自己写错了 抱歉
1 楼 502220545 2012-07-06  
您好 看了您写的例子 有一点小小的建议
1 当放数据的睡眠时间 大于 取数据的睡眠时间时 就会跑出异常

可以将 arrayBlockingQueue的 add和remove方法 换成 put 和 take 这样就不会报异常了 是以阻塞的形式运行的

相关推荐

    java多线程加队列上传文件_后台处理

    ### Java多线程加队列上传文件_后台处理 #### 概述 本文将详细介绍一个基于Java实现的多线程文件上传系统,并结合队列管理技术来优化后台处理流程。该系统通过创建多个线程来并行处理客户端的文件上传请求,同时...

    java多线程模拟队列实现排队叫号

    总的来说,通过Java多线程和队列数据结构,我们可以有效地模拟排队叫号系统。这种模拟有助于理解和实践并发编程,同时也为我们解决实际问题提供了思路。在这个过程中,我们学习了线程同步、队列操作以及如何在Java中...

    java多线程读取文件

    Java多线程读大文件 java多线程写文件:多线程往队列中写入数据

    支持多线程和泛型的阻塞队列

    阻塞队列是一种在多线程编程中广泛使用的并发数据结构,它在计算机科学和编程领域,特别是Java和C++等面向对象语言中扮演着重要角色。标题中的“支持多线程和泛型的阻塞队列”意味着我们讨论的是一个能够同时处理多...

    java模拟阻塞队列

    Java中的阻塞队列是一种基于同步原语的高级数据结构,它在多线程编程中扮演着重要角色,尤其在并发处理和优化系统资源利用率方面。阻塞队列结合了队列的数据结构与线程同步机制,使得生产者可以在队列满时被阻塞,而...

    idea+java多线程模拟队列实现排队叫号.zip

    在本项目"idea+java多线程模拟队列实现排队叫号.zip"中,我们探讨的是如何使用Java来模拟一个基于多线程的排队叫号系统。IntelliJ IDEA(简称Idea)是一款强大的Java集成开发环境,它为编写、调试和优化Java代码提供...

    java多线程经典案例

    本案例将深入探讨Java多线程中的关键知识点,包括线程同步、线程通信和线程阻塞。 线程同步是为了防止多个线程同时访问共享资源,导致数据不一致。Java提供了多种同步机制,如synchronized关键字、Lock接口...

    Java多线程调用BlockingDeque跑批量数据的例子

    一个线程从A表读数据放入队列 N个线程从队列中取出数据,找到其在子表中的数据 对子表中的数据开3种线程:读,发email,标记状态 N个线程对完成的A表数据做最后处理 支持大数据量跑批,就是个例子,本来是公司发送...

    易语言简单的多线程消息队列

    易语言简单的多线程消息队列。@Patek。

    Java多线程知识点总结

    Java多线程是Java编程语言中一个非常重要的概念,它允许开发者在一个程序中创建多个执行线程并行运行,以提高程序的执行效率和响应速度。在Java中,线程的生命周期包含五个基本状态,分别是新建状态(New)、就绪...

    秒杀多线程第十六篇 多线程十大经典案例之一 双线程读写队列数据

    《秒杀多线程第十六篇 多线程十大经典案例之一 双线程读写队列数据》 ...为了让大家更加熟练运用多线程,将会有十篇文章来讲解十个多线程使用案例,相信看完这十篇后会让你能更加游刃有余的使用多线程。

    java多线程的讲解和实战

    Java多线程是Java编程中的重要概念,尤其在如今的多核处理器环境下,理解并熟练掌握多线程技术对于提高程序性能和响应速度至关重要。本资料详细讲解了Java多线程的原理,并提供了丰富的实战代码,非常适合Java初学者...

    java多线程解决消息压入栈和取出

    下面是一个简单的示例,展示如何使用Java多线程和消息队列: ```java import java.util.concurrent.BlockingQueue; import java.util.concurrent.LinkedBlockingQueue; public class MessageQueueExample { ...

    java多线程实现大批量数据导入源码

    在Java编程中,多线程技术是处理大数据批量导入或导出的重要手段。它能有效提升程序执行效率,尤其在数据库操作这样的I/O密集型任务中。本项目以"java多线程实现大批量数据导入源码"为题,旨在通过多线程策略将大量...

    Java多线程编程实战指南-核心篇

    Java的BlockingQueue阻塞队列可以实现线程间的生产者-消费者模式,通过put()和take()方法实现数据的传递。此外,Phaser、CyclicBarrier和CountDownLatch等同步辅助类也是线程间协调的重要工具。 线程池是Java并发...

    java 多线程编程实战指南(核心 + 设计模式 完整版)

    《Java多线程编程实战指南》这本书深入浅出地讲解了Java多线程的核心概念和实战技巧,分为核心篇和设计模式篇,旨在帮助开发者掌握并应用多线程技术。 1. **线程基础** - **线程的创建**:Java提供了两种创建线程...

    多线程 队列利用

    在IT行业中,多线程和队列是两个关键的概念,特别是在并发编程和高效系统设计中。本篇文章将深入探讨这两个概念及其结合应用。...通过合理使用多线程和队列,开发者可以构建出更加健壮和高效的软件系统。

    多线程任务队列

    根据提供的文件名"Deque_MT.h",我们可以猜测这是一个包含多线程任务队列实现的头文件,其中可能使用了`std::deque`作为底层数据结构。`std::deque`(双端队列)是C++标准库中的一个容器,支持在两端高效地插入和...

Global site tag (gtag.js) - Google Analytics