`
yonlist
  • 浏览: 85372 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

Java 阻塞队列

 
阅读更多

项目中需要用到多线程处理,线程的任务最好是动态的负载均衡,自然想到用阻塞队列。再次学习:

 

==============================================================================

 

Queue接口与List、Set同一级别,都是继承了Collection接口。LinkedList实现了Queue接口。Queue接口窄化了对LinkedList的方法的访问权限(即在方法中的参数类型如果是Queue时,就完全只能访问Queue接口所定义的方法了,而不能直接访问LinkedList的非Queue的方法),以使得只有恰当的方法才可以使用。BlockingQueue 继承了Queue接口。

 

队列是一种数据结构.它有两个基本操作:在队列尾部加人一个元素,和从队列头部移除一个元素就是说,队列以一种先进先出的方式管理数据,如果你试图向一个已经满了的阻塞队列中添加一个元素或者是从一个空的阻塞队列中移除一个元索,将导致线程阻塞.在多线程进行合作时,阻塞队列是很有用的工具。工作者线程可以定期地把中间结果存到阻塞队列中而其他工作者线线程把中间结果取出并在将来修改它们。队列会自动平衡负载。如果第一个线程集运行得比第二个慢,则第二个线程集在等待结果时就会阻塞。如果第一个线程集运行得快,那么它将等待第二个线程集赶上来。下表显示了jdk1.5中的阻塞队列的操作:

 

add        增加一个元索                     如果队列已满,则抛出一个IIIegaISlabEepeplian异常
remove   移除并返回队列头部的元素    如果队列为空,则抛出一个NoSuchElementException异常
element  返回队列头部的元素             如果队列为空,则抛出一个NoSuchElementException异常
offer       添加一个元素并返回true       如果队列已满,则返回false
poll         移除并返问队列头部的元素    如果队列为空,则返回null
peek       返回队列头部的元素             如果队列为空,则返回null
put         添加一个元素                      如果队列满,则阻塞
take        移除并返回队列头部的元素     如果队列为空,则阻塞

 

remove、element、offer 、poll、peek 其实是属于Queue接口。 

 

阻塞队列的操作可以根据它们的响应方式分为以下三类:aad、removee和element操作在你试图为一个已满的队列增加元素或从空队列取得元素时抛出异常。当然,在多线程程序中,队列在任何时间都可能变成满的或空的,所以你可能想使用offer、poll、peek方法。这些方法在无法完成任务时只是给出一个出错示而不会抛出异常。

 

注意:poll和peek方法出错进返回null。因此,向队列中插入null值是不合法的。

 

还有带超时的offer和poll方法变种,例如,下面的调用:
boolean success = q.offer(x,100,TimeUnit.MILLISECONDS);
尝试在100毫秒内向队列尾部插入一个元素。如果成功,立即返回true;否则,当到达超时进,返回false。同样地,调用:
Object head = q.poll(100, TimeUnit.MILLISECONDS);
如果在100毫秒内成功地移除了队列头元素,则立即返回头元素;否则在到达超时时,返回null。

 

最后,我们有阻塞操作put和take。put方法在队列满时阻塞,take方法在队列空时阻塞。

 

java.ulil.concurrent包提供了阻塞队列的4个变种。默认情况下,LinkedBlockingQueue的容量是没有上限的(说的不准确,在不指定时容量为Integer.MAX_VALUE,不要然的话在put时怎么会受阻呢),但是也可以选择指定其最大容量,它是基于链表的队列,此队列按 FIFO(先进先出)排序元素。


ArrayBlockingQueue在构造时需要指定容量,并可以选择是否需要公平性,如果公平参数被设置true,等待时间最长的线程会优先得到处理(其实就是通过将ReentrantLock设置为true来达到这种公平性的:即等待时间最长的线程会先操作)。通常,公平性会使你在性能上付出代价,只有在的确非常需要的时候再使用它。它是基于数组的阻塞循环队列,此队列按 FIFO(先进先出)原则对元素进行排序。


PriorityBlockingQueue是一个带优先级的队列,而不是先进先出队列。元素按优先级顺序被移除,该队列也没有上限(看了一下源码,PriorityBlockingQueue是对PriorityQueue的再次包装,是基于堆数据结构的,而PriorityQueue是没有容量限制的,与ArrayList一样,所以在优先阻塞队列上put时是不会受阻的。虽然此队列逻辑上是无界的,但是由于资源被耗尽,所以试图执行添加操作可能会导致 OutOfMemoryError),但是如果队列为空,那么取元素的操作take就会阻塞,所以它的检索操作take是受阻的。另外,往入该队列中的元素要具有比较能力。


最后,DelayQueue(基于PriorityQueue来实现的)是一个存放Delayed 元素的无界阻塞队列,只有在延迟期满时才能从中提取元素。该队列的头部是延迟期满后保存时间最长的 Delayed 元素。如果延迟都还没有期满,则队列没有头部,并且poll将返回null。当一个元素的 getDelay(TimeUnit.NANOSECONDS) 方法返回一个小于或等于零的值时,则出现期满,poll就以移除这个元素了。此队列不允许使用 null 元素。 下面是延迟接口:

Java代码   收藏代码
  1. public interface Delayed extends Comparable<Delayed> {  
  2.      long getDelay(TimeUnit unit);  
  3. }  

放入DelayQueue的元素还将要实现compareTo方法,DelayQueue使用这个来为元素排序。

 

下面的实例展示了如何使用阻塞队列来控制线程集。程序在一个目录及它的所有子目录下搜索所有文件,打印出包含指定关键字的文件列表。从下面实例可以看出,使用阻塞队列两个显著的好处就是:多线程操作共同的队列时不需要额外的同步,另外就是队列会自动平衡负载,即那边(生产与消费两边)处理快了就会被阻塞掉,从而减少两边的处理速度差距。下面是具体实现:

Java代码   收藏代码
  1. public class BlockingQueueTest {  
  2.     public static void main(String[] args) {  
  3.         Scanner in = new Scanner(System.in);  
  4.         System.out.print("Enter base directory (e.g. /usr/local/jdk5.0/src): ");  
  5.         String directory = in.nextLine();  
  6.         System.out.print("Enter keyword (e.g. volatile): ");  
  7.         String keyword = in.nextLine();  
  8.   
  9.         final int FILE_QUEUE_SIZE = 10;// 阻塞队列大小  
  10.         final int SEARCH_THREADS = 100;// 关键字搜索线程个数  
  11.   
  12.         // 基于ArrayBlockingQueue的阻塞队列  
  13.         BlockingQueue<File> queue = new ArrayBlockingQueue<File>(  
  14.                 FILE_QUEUE_SIZE);  
  15.   
  16.         //只启动一个线程来搜索目录  
  17.         FileEnumerationTask enumerator = new FileEnumerationTask(queue,  
  18.                 new File(directory));  
  19.         new Thread(enumerator).start();  
  20.           
  21.         //启动100个线程用来在文件中搜索指定的关键字  
  22.         for (int i = 1; i <= SEARCH_THREADS; i++)  
  23.             new Thread(new SearchTask(queue, keyword)).start();  
  24.     }  
  25. }  
  26. class FileEnumerationTask implements Runnable {  
  27.     //哑元文件对象,放在阻塞队列最后,用来标示文件已被遍历完  
  28.     public static File DUMMY = new File("");  
  29.   
  30.     private BlockingQueue<File> queue;  
  31.     private File startingDirectory;  
  32.   
  33.     public FileEnumerationTask(BlockingQueue<File> queue, File startingDirectory) {  
  34.         this.queue = queue;  
  35.         this.startingDirectory = startingDirectory;  
  36.     }  
  37.   
  38.     public void run() {  
  39.         try {  
  40.             enumerate(startingDirectory);  
  41.             queue.put(DUMMY);//执行到这里说明指定的目录下文件已被遍历完  
  42.         } catch (InterruptedException e) {  
  43.         }  
  44.     }  
  45.   
  46.     // 将指定目录下的所有文件以File对象的形式放入阻塞队列中  
  47.     public void enumerate(File directory) throws InterruptedException {  
  48.         File[] files = directory.listFiles();  
  49.         for (File file : files) {  
  50.             if (file.isDirectory())  
  51.                 enumerate(file);  
  52.             else  
  53.                 //将元素放入队尾,如果队列满,则阻塞  
  54.                 queue.put(file);  
  55.         }  
  56.     }  
  57. }  
  58. class SearchTask implements Runnable {  
  59.     private BlockingQueue<File> queue;  
  60.     private String keyword;  
  61.   
  62.     public SearchTask(BlockingQueue<File> queue, String keyword) {  
  63.         this.queue = queue;  
  64.         this.keyword = keyword;  
  65.     }  
  66.   
  67.     public void run() {  
  68.         try {  
  69.             boolean done = false;  
  70.             while (!done) {  
  71.                 //取出队首元素,如果队列为空,则阻塞  
  72.                 File file = queue.take();  
  73.                 if (file == FileEnumerationTask.DUMMY) {  
  74.                     //取出来后重新放入,好让其他线程读到它时也很快的结束  
  75.                     queue.put(file);  
  76.                     done = true;  
  77.                 } else  
  78.                     search(file);  
  79.             }  
  80.         } catch (IOException e) {  
  81.             e.printStackTrace();  
  82.         } catch (InterruptedException e) {  
  83.         }  
  84.     }  
  85.     public void search(File file) throws IOException {  
  86.         Scanner in = new Scanner(new FileInputStream(file));  
  87.         int lineNumber = 0;  
  88.         while (in.hasNextLine()) {  
  89.             lineNumber++;  
  90.             String line = in.nextLine();  
  91.             if (line.contains(keyword))  
  92.                 System.out.printf("%s:%d:%s%n", file.getPath(), lineNumber,  
  93.                         line);  
  94.         }  
  95.         in.close();  
  96.     }  
  97. }  

 

==============================================================================

以上内容转载自:http://jiangzhengjun.iteye.com/blog/683593

分享到:
评论

相关推荐

    阻塞队列阻塞队列阻塞队列

    在Java编程语言中,阻塞队列是一种线程安全的数据结构,它在多线程并发控制中发挥着重要作用。阻塞队列的核心特性是当队列为空时,尝试获取元素的线程会被阻塞,直到其他线程添加元素;同样,当队列满时,试图插入...

    java阻塞队列实现原理及实例解析.docx

    总结来说,Java阻塞队列是一种强大的并发工具,它通过自动阻塞和唤醒线程来实现线程间的同步,简化了多线程编程的复杂性。在Java `java.util.concurrent`包中提供了多种优化的阻塞队列实现,适用于不同的并发需求。...

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

    通过以上分析,我们了解到基于Java阻塞队列的搜索实例利用了Java并发API的强大功能来实现多线程间的高效协作,从而完成复杂的文件搜索任务。在多线程编程实践中,理解阻塞队列的工作原理和正确使用它们是关键的一步...

    消息分发框架(基于JAVA阻塞队列实现、 生产者消费者模型)

    Java阻塞队列是并发编程中的重要工具,它是一个线程安全的数据结构,提供了在生产者和消费者之间同步共享数据的方法。阻塞队列在满时会阻止生产者继续插入元素,在空时会阻止消费者进行消费。这种机制可以防止数据...

    java阻塞队列实现原理及实例解析

    Java阻塞队列实现原理及实例解析 Java阻塞队列是一种特殊的队列,它能够在队列为空或满时阻塞线程,使得线程之间能够更好地协作和通信。阻塞队列的实现原理是基于锁机制和条件变量机制的,通过wait和notify方法来...

    Java阻塞队列四组API介绍(小结)

    Java阻塞队列四组API介绍 Java阻塞队列四组API介绍是Java并发编程中非常重要的一部分,本文将详细介绍这四组API,并通过示例代码进行讲解。 第一组API:会抛出异常的API 第一组API包括add()、remove()、element()...

    java模拟阻塞队列

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

    详解Java阻塞队列(BlockingQueue)的实现原理

    "详解Java阻塞队列(BlockingQueue)的实现原理" Java阻塞队列(BlockingQueue)是Java.util.concurrent包下重要的数据结构,提供了线程安全的队列访问方式。BlockingQueue的实现原理主要是基于四组不同的方法用于...

    剖析Java中阻塞队列的实现原理及应用场景

    Java中的阻塞队列是一种特殊的线程安全的数据结构,它在多线程环境下用于高效地处理生产者-消费者问题。阻塞队列的核心特性在于当队列为空时,尝试获取元素的线程会被阻塞,直到队列中有元素可用;同样,当队列满时...

    Java实现简单的阻塞队列2种方式

    在Java编程中,阻塞队列是一种特殊类型的并发数据结构,它在多线程环境中的应用广泛,主要用于线程间的协作通信。阻塞队列在队列满时会阻止生产者线程添加元素,在队列空时会阻止消费者线程取出元素,直到条件满足...

    Java 阻塞队列详解及简单使用

    Java 阻塞队列(BlockingQueue)...总的来说,Java阻塞队列是一个强大的工具,它简化了多线程环境下的数据共享和线程同步,提高了系统的效率和可维护性。了解并熟练使用阻塞队列,对于编写高性能并发应用程序至关重要。

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

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

    14-阻塞队列BlockingQueue实战及其原理分析二.pdf

    阻塞队列(BlockingQueue)是一种特殊的队列,它支持两个附加操作:阻塞的插入方法put和阻塞的移除方法take。BlockingQueue继承了Queue接口,是Java 5中加入的。 BlockingQueue常用方法示例: 1. add(E e):添加一...

    java线程聊天室(阻塞队列实现)

    【Java线程聊天室(阻塞队列实现)】 在Java编程中,多线程是构建并发应用程序的关键技术。在创建一个线程聊天室时,我们通常会涉及到多个线程之间的交互,例如用户发送消息、接收消息以及处理网络通信等。而阻塞...

    java 中 阻塞队列BlockingQueue详解及实例

    Java中的阻塞队列BlockingQueue是一种并发编程中常用的工具,它实现了线程间的同步和通信。阻塞队列的核心特性在于当队列为空时,尝试获取元素的线程会被阻塞,直到其他线程添加元素;当队列满时,尝试添加元素的...

    并发-线程池和阻塞队列.pdf

    Java中的ArrayBlockingQueue和LinkedBlockingQueue都是典型的阻塞队列实现。 阻塞队列为线程间通信提供了便捷的机制,可以用来协调多个线程的运行,防止多线程直接访问共享资源导致的并发问题。例如,生产者-消费者...

    BlockingQueue(阻塞队列)详解

    ### BlockingQueue(阻塞队列)详解 #### 一、前言 随着现代软件系统对并发性能需求的不断提高,多线程编程技术逐渐成为开发人员不可或缺的技能之一。在Java平台中,`java.util.concurrent`包提供了丰富的工具来...

    阻塞队列实现生产者消费者模式Java开发Java经验技巧共

    在"阻塞队列实现生产者消费者模式Java开发Java经验技巧共4页.pdf.zip"这个压缩包中,很可能是详细介绍了如何使用Java的阻塞队列来构建生产者消费者模式,可能包括以下知识点: 1. **阻塞队列接口**:首先,会介绍`...

    并发-线程池和阻塞队列

    在Java编程中,"并发-线程池和阻塞队列"是两个核心概念,它们在多线程环境下处理任务调度和数据同步方面发挥着重要作用。线程池是一种管理线程资源的有效方式,而阻塞队列则常用于线程间通信和数据共享。 线程池...

Global site tag (gtag.js) - Google Analytics