java.util.concurrent 包里的 BlockingQueue 接口表示一个线程安放入和提取实例的队列。
BlockingQueue用法
BlockingQueue 通常用于一个线程生产对象,而另外一个线程消费这些对象的场景。下图是
对这个原理的阐述:
一个线程往里边放,另外一个线程从里边取的一个 BlockingQueue 。
一个线程将会持续生产新对象并将其插入到队列之中,直到队列达到它所能容纳的临界点。
也就是说,它是有限的。如果该阻塞队列到达了其临界点,负责生产的线程将会在往里边插
入新对象时发生阻塞。它会一直处于阻塞之中,直到负责消费的线程从队列中拿走一个对象。
负责消费的线程将会一直从该阻塞队列中拿出对象。如果消费线程尝试去从一个空的队列中
提取对象的话,这个消费线程将会处于阻塞之中,直到一个生产线程把一个对象丢进队列。
BlockingQueue 的方法
BlockingQueue 具有 4 组不同的方法用于插入、移除以及对队列中的元素进行检查。如果
请求的操作不能得到立即执行的话,每个方法的表现也不同。这些方法如下:
四组不同的行为方式解释:
- 抛异常:如果试图的操作无法立即执行,抛一个异常。
- 特定值:如果试图的操作无法立即执行,返回一个特定的值(常常是 true / false)。
- 阻塞:如果试图的操作无法立即执行,该方法调用将会发生阻塞,直到能够执行。
- 超时:如果试图的操作无法立即执行,该方法调用将会发生阻塞,直到能够执行,但等
待时间不会超过给定值。返回一个特定值以告知该操作是否成功(典型的是 true / false)。
无法向一个 BlockingQueue 中插入 null。如果你试图插入 null,BlockingQueue 将会抛出
一个 NullPointerException。
可以访问到 BlockingQueue 中的所有元素,而不仅仅是开始和结束的元素。比如说,你将
一个对象放入队列之中以等待处理,但你的应用想要将其取消掉。那么你可以调用诸如
remove(o) 方法来将队列之中的特定对象进行移除。但是这么干效率并不高(译者注:基于队
列的数据结构,获取除开始或结束位置的其他对象的效率不会太高),因此你尽量不要用这
一类的方法,除非你确实不得不那么做。
BlockingQueue 的实现
BlockingQueue 是 个 接 口 , 你 需 要 使 用 它 的 实 现 之 一 来 使 用 BlockingQueue 。
java.util.concurrent 具有以下 BlockingQueue 接口的实现(Java 6):
- ArrayBlockingQueue 数组阻塞队列
- DelayQueue 延迟队列
- LinkedBlockingQueue 链阻塞队列
- PriorityBlockingQueue 具有优先级的阻塞队列
- SynchronousQueue 同步队列