在工作环境中,我们通常会使用ArrayBlockingQueue 作为任务管理队列,并且之前在Qunar开发的时候,也用过ArrayBlockingQueue作为异步队列,实现了一个异步通知的程序。最近有点时间,阅读一下这个工具。
我们知道ArrayBlockingQueue是一个FIFO的有界阻塞式对列,每次取对列的元素的时候,取的是队首的元素,而每次往这个对列的队尾放元素。
/** The queued items */
final Object[] items;
/** items index for next take, poll, peek or remove */
int takeIndex;
/** items index for next put, offer, or add */
int putIndex;
/** Number of elements in the queue */
int count;
/*
* Concurrency control uses the classic two-condition algorithm
* found in any textbook.
*/
/** Main lock guarding all access */
final ReentrantLock lock;
/** Condition for waiting takes */
private final Condition notEmpty;
/** Condition for waiting puts */
private final Condition notFull;
/**
* Shared state for currently active iterators, or null if there
* are known not to be any. Allows queue operations to update
* iterator state.
*/
transient Itrs itrs = null;
items 这个属性是用来保存队列里面的元素的,因为这个队列叫做Array,所以元素用数组保存。
takeIndex 这个属性是标明每次取出来的元素位置,这个可以理解为队首。
putIndex 这个属性标明每次放的位置,可以理解成队尾。
count 这个属性表示队列中元素的数量,这样保存有个好处,就不用每次都去遍历,数出元素个数。
lock 这个是锁,用来防止并发操作。
notEmpty 取出元素的等待条件。
notFull 往队列中放元素的等待条件。
itrs 因为这个queue实现了Iterable 接口,该属性保存了迭代器的状态。
2. 方法源码阅读
属性介绍完了,我们来看一看这个类包含的方法,以及方法的具体实现。
/**
* Creates an {@code ArrayBlockingQueue} with the given (fixed)
* capacity and the specified access policy.
*
* @param capacity the capacity of this queue
* @param fair if {@code true} then queue accesses for threads blocked
* on insertion or removal, are processed in FIFO order;
* if {@code false} the access order is unspecified.
* @throws IllegalArgumentException if {@code capacity < 1}
*/
public ArrayBlockingQueue(int capacity, boolean fair) {
if (capacity <= 0)
throw new IllegalArgumentException();
this.items = new Object[capacity];
lock = new ReentrantLock(fair);
notEmpty = lock.newCondition();
notFull = lock.newCondition();
}
/**
* Creates an {@code ArrayBlockingQueue} with the given (fixed)
* capacity, the specified access policy and initially containing the
* elements of the given collection,
* added in traversal order of the collection's iterator.
*
* @param capacity the capacity of this queue
* @param fair if {@code true} then queue accesses for threads blocked
* on insertion or removal, are processed in FIFO order;
* if {@code false} the access order is unspecified.
* @param c the collection of elements to initially contain
* @throws IllegalArgumentException if {@code capacity} is less than
* {@code c.size()}, or less than 1.
* @throws NullPointerException if the specified collection or any
* of its elements are null
*/
public ArrayBlockingQueue(int capacity, boolean fair,
Collection extends E> c) {
this(capacity, fair);
final ReentrantLock lock = this.lock;
lock.lock(); // Lock only for visibility, not mutual exclusion
try {
int i = 0;
try {
for (E e : c) {
checkNotNull(e);
items[i++] = e;
}
} catch (ArrayIndexOutOfBoundsException ex) {
throw new IllegalArgumentException();
}
count = i;
putIndex = (i == capacity) ? 0 : i;
} finally {
lock.unlock();
}
}
这个类提供了三个构造函数,我这里只列出了两个,因为有一个就是调用了我列的第二个。
首先看参数列表为int capacity, boolean fair
这个构造函数只是初始化了数组,两个等待条件,还有用来同步的锁,这个锁可以选择是否是公平锁还是非公平锁。
再看另一个构造函数可以传递一个Collection的构造函数。
该构造函数可以以一个现有的实现了Collection接口的元素为基础创建队列,不过,这里使用了ReentrantLock来锁住代码块,是不是很奇怪?注释里面说,这个锁只是用来保证可见性,并不是用来互斥的。并且,该方法catch住了ArrayIndexOutOfBoundsException 异常,这是因为,c是由外部传进来的,构造函数并不能保证c不被其他线程修改,所以有可能会在遍历的过程中抛ArrayIndexOutOfBoundsException(具体可以参考《java 并发编程实战》)
往下就是我们平时最关心的几个方法了
add(E e) 当队列不满,调用offer方法,当队列已满,抛出IllegalStateException 异常
offer(E e) 当队列不满,将元素放进队列,并返回成功,当队列满了,立刻返回false
put(E e) 阻塞式插入,如果队列元素满,会阻塞到队列有空间
offer(E e, long timeout, TimeUnit unit) 等待timeout时间的offer
这些方法都是往队列里面放元素的操作,但是它们有一些细微的不同,我们可以来看一下源码
public boolean offer(E e) {
checkNotNull(e);
final ReentrantLock lock = this.lock;
lock.lock();
try {
if (count == items.length)
return false;
else {
enqueue(e);
return true;
}
} finally {
lock.unlock();
}
}
先看offer操作,该操作的作用是,如果队列满了,会立刻返回false,如果队列没满,将元素插入队列,此方法调用了checkNotNull方法,在传递进来的元素为空会抛出NullPointerException异常。
这个方法使用了对象的锁,将队列长度的校验和插入操作锁起来,避免了并发问题。
再看add操作
public boolean add(E e) {
return super.add(e);
}
/**将超类的代码贴在下面*/
public boolean add(E e) {
if (offer(e))
return true;
else
throw new IllegalStateException("Queue full");
}
从add方法代码可以看出,这个方法会先去调用offer ,如果返回false ,直接抛出异常。
下面接着看put方法,这个是我认为BlockingQueue提供的核心方法了,即阻塞的插入队列,之前和我们组的一个应届生讨论,他觉得这个阻塞应该是自旋式的阻塞,但是我说看一下代码,发现其实用的是Condition类的await,这个方法最后使用了LockSupport.unpark原语进行操作(有空可以研究一下Condition的代码)
public void put(E e) throws InterruptedException {
checkNotNull(e);
final ReentrantLock lock = this.lock;
lock.lockInterruptibly();
try {
while (count == items.length)
notFull.await();
enqueue(e);
} finally {
lock.unlock();
}
}
最后再看带有等待时间的offer,该方法调用了Condition的awaitNanos ,相当于await一定的时间。
public boolean offer(E e, long timeout, TimeUnit unit)
throws InterruptedException {
checkNotNull(e);
long nanos = unit.toNanos(timeout);
final ReentrantLock lock = this.lock;
lock.lockInterruptibly();
try {
while (count == items.length) {
if (nanos <= 0)
return false;
nanos = notFull.awaitNanos(nanos);
}
enqueue(e);
return true;
} finally {
lock.unlock();
}
}
以上几个往队列里面插入元素的方法,最后都会调用到,这个方法很简单,就是在putIndex位置插入元素,但是最后会使用notEmpty的signal,因为插入之后,队列不为空,需要通知阻塞在take等方法的线程
private void enqueue(E x) {
// assert lock.getHoldCount() == 1;
// assert items[putIndex] == null;
final Object[] items = this.items;
items[putIndex] = x;
if (++putIndex == items.length)
putIndex = 0;
count++;
notEmpty.signal();
}