java中使用队列:java.util.Queue (转)

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

转载自:http://www.cnblogs.com/end/archive/2012/10/25/2738493.html

时间: 2024-08-07 21:18:55

java中使用队列:java.util.Queue (转)的相关文章

Java中数组操作 java.util.Arrays 类常用方法的使用

任何一门编程语言,数组都是最重要和常用的数据结构之一,但不同的语言对数组的构造与处理是不尽相同的. Java中提供了java.util.Arrays 类能方便地操作数组,并且它提供的所有方法都是静态的.下面介绍一下Arrays类最常用的几个方法. 1.  数组排序 Arrays工具类提供了一个sort方法,只需要一行代码即可完成排序功能. 2.  数组转换为字符串 Arrays提供了一个toString方法,可以直接把一个数组转换为字符串,这样可以方便观察数组里的元素. //来源:公众号[时光与

Java中的队列API——Queue

队列具有FIFO(先进先出)的功能,那么如何实现这一功能呢,呵呵呵,Java已经为我们提供了API--Queue,下面通过一个例子来认识一下该API: import java.util.Queue; import java.util.LinkedList; public class TestQueue { public static void main(String[] args) { Queue<String> queue = new LinkedList<String>();

Java中阻塞队列的使用

http://blog.csdn.net/qq_35101189/article/details/56008342 在新增的Concurrent包中,BlockingQueue很好的解决了多线程中,如何高效安全“传输”数据的问题.通过这些高效并且线程安全的队列类,为我们快速搭建高质量的多线程程序带来极大的便利.本文详细介绍了BlockingQueue家庭中的所有成员,包括他们各自的功能以及常见使用场景. 认识BlockingQueue阻塞队列,顾名思义,首先它是一个队列,而一个队列在数据结构中所

Java中的日志——Java.util.logging、log4j、commons-logging

Java中给项目程序添加log主要有三种方式,一使用JDK中的java.util.logging包,一种是log4j,一种是commons-logging.其中log4j和commons-logging都是apache软件基金会的开源项目.这三种方式的区别如下: Java.util.logging,JDK标准库中的类,是JDK 1.4 版本之后添加的日志记录的功能包. log4j,最强大的记录日志的方式.可以通过配置 .properties 或是 .xml 的文件, 配置日志的目的地,格式等等.

Java中SQL DATE和 UTIL DATE 的相互转换

public String getStringtime (java.sql.Date sdate ){ java.util.Date date = new java.util.Date(sdate.getTime()); SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd"); String time =sdf.format(date); return time; } public java.sql.Date getSdate

Java中的泛型 --- Java 编程思想

前言 ? 我一直都认为泛型是程序语言设计中一个非常基础,重要的概念,Java 中的泛型到底是怎么样的,为什么会有泛型,泛型怎么发展出来的.通透理解泛型是学好基础里面中非常重要的.于是,我对<Java编程思想>这本书中泛型章节进行了研读.可惜遗憾的是,自己没有太多的经验,有些东西看了几次也是有点懵.只能以后有机会,再进行学习了.但是自己也理解了挺多的.下面就是自己对于泛型的理解与感悟.如有不对,望指出. 概念 由来: Java 一开始设计之初是没有泛型这个特性的,直到jdk 1.5中引入了这个特

避免Java中NullPointerException的Java技巧和最佳实践

Java中的NullPointerException是我们最经常遇到的异常了,那我们到底应该如何在编写代码是防患于未然呢.下面我们就从几个方面来入手,解决这个棘手的?问题吧.? 值得庆幸的是,通过应用一些防御性编码技术并遵循应用程序多个部分之间的约定,您可以在一定程度上避免Java中的NullPointerException. 顺便说一下,在本文中,我们将学习一些Java的编码技术和最佳实践,这些技巧和最佳实践可用于避免的Java中的空指针异常.遵循这些Java的技巧还可以最大程度地减少很多Ja

java中的队列Queue

一.概述 其位于java.util包下,声明:public interface Queue<E> extends Collection<E> 在处理元素前用于保存元素的 collection.除了基本的 Collection 操作外,队列还提供其他的插入.提取和检查操作.每个方法都存在两种形式:一种抛出异常(操作失败时),另一种返回一个特殊值(null 或 false,具体取决于操作).插入操作的后一种形式是用于专门为有容量限制的 Queue 实现设计的:在大多数实现中,插入操作不

java中的队列

非阻塞队列:ConcurrentLinkedQueue ConcurrentLinkedQueue是一个基于链接节点的无界线程安全队列,它采用先进先出的规则对节点进行排序,当我们添加一个元素的时候,它会添加到队列的尾部,当我们获取一个元素时,它会返回队列头部的元素. 阻塞队列:BlockingQueue 1. ArrayBlockingQueue      基于数组的阻塞队列实现,在ArrayBlockingQueue内部,维护了一个定长数组,以便缓存队列中的数据对象,这是一个常用的阻塞队列,除