1 /*
2  * Written by Doug Lea with assistance from members of JCP JSR-166
3  * Expert Group and released to the public domain, as explained at
4  * http://creativecommons.org/publicdomain/zero/1.0/
5  * Other contributors include Andrew Wright, Jeffrey Hayes,
6  * Pat Fisher, Mike Judd.
7  */
8 
9 package jsr166;
10 
11 import java.util.Arrays;
12 import java.util.Collection;
13 import java.util.Iterator;
14 import java.util.NoSuchElementException;
15 import java.util.Queue;
16 import java.util.concurrent.ConcurrentLinkedQueue;
17 
18 import junit.framework.Test;
19 import junit.framework.TestSuite;
20 
21 public class ConcurrentLinkedQueueTest extends JSR166TestCase {
22 
23     // android-note: Removed because the CTS runner does a bad job of
24     // retrying tests that have suite() declarations.
25     //
26     // public static void main(String[] args) {
27     //     main(suite(), args);
28     // }
29     // public static Test suite() {
30     //     return new TestSuite(ConcurrentLinkedQueueTest.class);
31     // }
32 
33     /**
34      * Returns a new queue of given size containing consecutive
35      * Integers 0 ... n.
36      */
37     private ConcurrentLinkedQueue<Integer> populatedQueue(int n) {
38         ConcurrentLinkedQueue<Integer> q = new ConcurrentLinkedQueue<Integer>();
39         assertTrue(q.isEmpty());
40         for (int i = 0; i < n; ++i)
41             assertTrue(q.offer(new Integer(i)));
42         assertFalse(q.isEmpty());
43         assertEquals(n, q.size());
44         return q;
45     }
46 
47     /**
48      * new queue is empty
49      */
50     public void testConstructor1() {
51         assertEquals(0, new ConcurrentLinkedQueue().size());
52     }
53 
54     /**
55      * Initializing from null Collection throws NPE
56      */
57     public void testConstructor3() {
58         try {
59             new ConcurrentLinkedQueue((Collection)null);
60             shouldThrow();
61         } catch (NullPointerException success) {}
62     }
63 
64     /**
65      * Initializing from Collection of null elements throws NPE
66      */
67     public void testConstructor4() {
68         try {
69             new ConcurrentLinkedQueue(Arrays.asList(new Integer[SIZE]));
70             shouldThrow();
71         } catch (NullPointerException success) {}
72     }
73 
74     /**
75      * Initializing from Collection with some null elements throws NPE
76      */
77     public void testConstructor5() {
78         Integer[] ints = new Integer[SIZE];
79         for (int i = 0; i < SIZE - 1; ++i)
80             ints[i] = new Integer(i);
81         try {
82             new ConcurrentLinkedQueue(Arrays.asList(ints));
83             shouldThrow();
84         } catch (NullPointerException success) {}
85     }
86 
87     /**
88      * Queue contains all elements of collection used to initialize
89      */
90     public void testConstructor6() {
91         Integer[] ints = new Integer[SIZE];
92         for (int i = 0; i < SIZE; ++i)
93             ints[i] = new Integer(i);
94         ConcurrentLinkedQueue q = new ConcurrentLinkedQueue(Arrays.asList(ints));
95         for (int i = 0; i < SIZE; ++i)
96             assertEquals(ints[i], q.poll());
97     }
98 
99     /**
100      * isEmpty is true before add, false after
101      */
102     public void testEmpty() {
103         ConcurrentLinkedQueue q = new ConcurrentLinkedQueue();
104         assertTrue(q.isEmpty());
105         q.add(one);
106         assertFalse(q.isEmpty());
107         q.add(two);
108         q.remove();
109         q.remove();
110         assertTrue(q.isEmpty());
111     }
112 
113     /**
114      * size changes when elements added and removed
115      */
116     public void testSize() {
117         ConcurrentLinkedQueue q = populatedQueue(SIZE);
118         for (int i = 0; i < SIZE; ++i) {
119             assertEquals(SIZE - i, q.size());
120             q.remove();
121         }
122         for (int i = 0; i < SIZE; ++i) {
123             assertEquals(i, q.size());
124             q.add(new Integer(i));
125         }
126     }
127 
128     /**
129      * offer(null) throws NPE
130      */
131     public void testOfferNull() {
132         ConcurrentLinkedQueue q = new ConcurrentLinkedQueue();
133         try {
134             q.offer(null);
135             shouldThrow();
136         } catch (NullPointerException success) {}
137     }
138 
139     /**
140      * add(null) throws NPE
141      */
142     public void testAddNull() {
143         ConcurrentLinkedQueue q = new ConcurrentLinkedQueue();
144         try {
145             q.add(null);
146             shouldThrow();
147         } catch (NullPointerException success) {}
148     }
149 
150     /**
151      * Offer returns true
152      */
153     public void testOffer() {
154         ConcurrentLinkedQueue q = new ConcurrentLinkedQueue();
155         assertTrue(q.offer(zero));
156         assertTrue(q.offer(one));
157     }
158 
159     /**
160      * add returns true
161      */
162     public void testAdd() {
163         ConcurrentLinkedQueue q = new ConcurrentLinkedQueue();
164         for (int i = 0; i < SIZE; ++i) {
165             assertEquals(i, q.size());
166             assertTrue(q.add(new Integer(i)));
167         }
168     }
169 
170     /**
171      * addAll(null) throws NPE
172      */
173     public void testAddAll1() {
174         ConcurrentLinkedQueue q = new ConcurrentLinkedQueue();
175         try {
176             q.addAll(null);
177             shouldThrow();
178         } catch (NullPointerException success) {}
179     }
180 
181     /**
182      * addAll(this) throws IAE
183      */
184     public void testAddAllSelf() {
185         ConcurrentLinkedQueue q = populatedQueue(SIZE);
186         try {
187             q.addAll(q);
188             shouldThrow();
189         } catch (IllegalArgumentException success) {}
190     }
191 
192     /**
193      * addAll of a collection with null elements throws NPE
194      */
195     public void testAddAll2() {
196         ConcurrentLinkedQueue q = new ConcurrentLinkedQueue();
197         try {
198             q.addAll(Arrays.asList(new Integer[SIZE]));
199             shouldThrow();
200         } catch (NullPointerException success) {}
201     }
202 
203     /**
204      * addAll of a collection with any null elements throws NPE after
205      * possibly adding some elements
206      */
207     public void testAddAll3() {
208         ConcurrentLinkedQueue q = new ConcurrentLinkedQueue();
209         Integer[] ints = new Integer[SIZE];
210         for (int i = 0; i < SIZE - 1; ++i)
211             ints[i] = new Integer(i);
212         try {
213             q.addAll(Arrays.asList(ints));
214             shouldThrow();
215         } catch (NullPointerException success) {}
216     }
217 
218     /**
219      * Queue contains all elements, in traversal order, of successful addAll
220      */
221     public void testAddAll5() {
222         Integer[] empty = new Integer[0];
223         Integer[] ints = new Integer[SIZE];
224         for (int i = 0; i < SIZE; ++i)
225             ints[i] = new Integer(i);
226         ConcurrentLinkedQueue q = new ConcurrentLinkedQueue();
227         assertFalse(q.addAll(Arrays.asList(empty)));
228         assertTrue(q.addAll(Arrays.asList(ints)));
229         for (int i = 0; i < SIZE; ++i)
230             assertEquals(ints[i], q.poll());
231     }
232 
233     /**
234      * poll succeeds unless empty
235      */
236     public void testPoll() {
237         ConcurrentLinkedQueue q = populatedQueue(SIZE);
238         for (int i = 0; i < SIZE; ++i) {
239             assertEquals(i, q.poll());
240         }
241         assertNull(q.poll());
242     }
243 
244     /**
245      * peek returns next element, or null if empty
246      */
247     public void testPeek() {
248         ConcurrentLinkedQueue q = populatedQueue(SIZE);
249         for (int i = 0; i < SIZE; ++i) {
250             assertEquals(i, q.peek());
251             assertEquals(i, q.poll());
252             assertTrue(q.peek() == null ||
253                        !q.peek().equals(i));
254         }
255         assertNull(q.peek());
256     }
257 
258     /**
259      * element returns next element, or throws NSEE if empty
260      */
261     public void testElement() {
262         ConcurrentLinkedQueue q = populatedQueue(SIZE);
263         for (int i = 0; i < SIZE; ++i) {
264             assertEquals(i, q.element());
265             assertEquals(i, q.poll());
266         }
267         try {
268             q.element();
269             shouldThrow();
270         } catch (NoSuchElementException success) {}
271     }
272 
273     /**
274      * remove removes next element, or throws NSEE if empty
275      */
276     public void testRemove() {
277         ConcurrentLinkedQueue q = populatedQueue(SIZE);
278         for (int i = 0; i < SIZE; ++i) {
279             assertEquals(i, q.remove());
280         }
281         try {
282             q.remove();
283             shouldThrow();
284         } catch (NoSuchElementException success) {}
285     }
286 
287     /**
288      * remove(x) removes x and returns true if present
289      */
290     public void testRemoveElement() {
291         ConcurrentLinkedQueue q = populatedQueue(SIZE);
292         for (int i = 1; i < SIZE; i += 2) {
293             assertTrue(q.contains(i));
294             assertTrue(q.remove(i));
295             assertFalse(q.contains(i));
296             assertTrue(q.contains(i - 1));
297         }
298         for (int i = 0; i < SIZE; i += 2) {
299             assertTrue(q.contains(i));
300             assertTrue(q.remove(i));
301             assertFalse(q.contains(i));
302             assertFalse(q.remove(i + 1));
303             assertFalse(q.contains(i + 1));
304         }
305         assertTrue(q.isEmpty());
306     }
307 
308     /**
309      * contains(x) reports true when elements added but not yet removed
310      */
311     public void testContains() {
312         ConcurrentLinkedQueue q = populatedQueue(SIZE);
313         for (int i = 0; i < SIZE; ++i) {
314             assertTrue(q.contains(new Integer(i)));
315             q.poll();
316             assertFalse(q.contains(new Integer(i)));
317         }
318     }
319 
320     /**
321      * clear removes all elements
322      */
323     public void testClear() {
324         ConcurrentLinkedQueue q = populatedQueue(SIZE);
325         q.clear();
326         assertTrue(q.isEmpty());
327         assertEquals(0, q.size());
328         q.add(one);
329         assertFalse(q.isEmpty());
330         q.clear();
331         assertTrue(q.isEmpty());
332     }
333 
334     /**
335      * containsAll(c) is true when c contains a subset of elements
336      */
337     public void testContainsAll() {
338         ConcurrentLinkedQueue q = populatedQueue(SIZE);
339         ConcurrentLinkedQueue p = new ConcurrentLinkedQueue();
340         for (int i = 0; i < SIZE; ++i) {
341             assertTrue(q.containsAll(p));
342             assertFalse(p.containsAll(q));
343             p.add(new Integer(i));
344         }
345         assertTrue(p.containsAll(q));
346     }
347 
348     /**
349      * retainAll(c) retains only those elements of c and reports true if change
350      */
351     public void testRetainAll() {
352         ConcurrentLinkedQueue q = populatedQueue(SIZE);
353         ConcurrentLinkedQueue p = populatedQueue(SIZE);
354         for (int i = 0; i < SIZE; ++i) {
355             boolean changed = q.retainAll(p);
356             if (i == 0)
357                 assertFalse(changed);
358             else
359                 assertTrue(changed);
360 
361             assertTrue(q.containsAll(p));
362             assertEquals(SIZE - i, q.size());
363             p.remove();
364         }
365     }
366 
367     /**
368      * removeAll(c) removes only those elements of c and reports true if changed
369      */
370     public void testRemoveAll() {
371         for (int i = 1; i < SIZE; ++i) {
372             ConcurrentLinkedQueue q = populatedQueue(SIZE);
373             ConcurrentLinkedQueue p = populatedQueue(i);
374             assertTrue(q.removeAll(p));
375             assertEquals(SIZE - i, q.size());
376             for (int j = 0; j < i; ++j) {
377                 Integer x = (Integer)(p.remove());
378                 assertFalse(q.contains(x));
379             }
380         }
381     }
382 
383     /**
384      * toArray contains all elements in FIFO order
385      */
386     public void testToArray() {
387         ConcurrentLinkedQueue q = populatedQueue(SIZE);
388         Object[] o = q.toArray();
389         for (int i = 0; i < o.length; i++)
390             assertSame(o[i], q.poll());
391     }
392 
393     /**
394      * toArray(a) contains all elements in FIFO order
395      */
396     public void testToArray2() {
397         ConcurrentLinkedQueue<Integer> q = populatedQueue(SIZE);
398         Integer[] ints = new Integer[SIZE];
399         Integer[] array = q.toArray(ints);
400         assertSame(ints, array);
401         for (int i = 0; i < ints.length; i++)
402             assertSame(ints[i], q.poll());
403     }
404 
405     /**
406      * toArray(null) throws NullPointerException
407      */
408     public void testToArray_NullArg() {
409         ConcurrentLinkedQueue q = populatedQueue(SIZE);
410         try {
411             q.toArray(null);
412             shouldThrow();
413         } catch (NullPointerException success) {}
414     }
415 
416     /**
417      * toArray(incompatible array type) throws ArrayStoreException
418      */
419     public void testToArray1_BadArg() {
420         ConcurrentLinkedQueue q = populatedQueue(SIZE);
421         try {
422             q.toArray(new String[10]);
423             shouldThrow();
424         } catch (ArrayStoreException success) {}
425     }
426 
427     /**
428      * iterator iterates through all elements
429      */
430     public void testIterator() {
431         ConcurrentLinkedQueue q = populatedQueue(SIZE);
432         Iterator it = q.iterator();
433         int i;
434         for (i = 0; it.hasNext(); i++)
435             assertTrue(q.contains(it.next()));
436         assertEquals(i, SIZE);
437         assertIteratorExhausted(it);
438     }
439 
440     /**
441      * iterator of empty collection has no elements
442      */
443     public void testEmptyIterator() {
444         assertIteratorExhausted(new ConcurrentLinkedQueue().iterator());
445     }
446 
447     /**
448      * iterator ordering is FIFO
449      */
450     public void testIteratorOrdering() {
451         final ConcurrentLinkedQueue q = new ConcurrentLinkedQueue();
452         q.add(one);
453         q.add(two);
454         q.add(three);
455 
456         int k = 0;
457         for (Iterator it = q.iterator(); it.hasNext();) {
458             assertEquals(++k, it.next());
459         }
460 
461         assertEquals(3, k);
462     }
463 
464     /**
465      * Modifications do not cause iterators to fail
466      */
467     public void testWeaklyConsistentIteration() {
468         final ConcurrentLinkedQueue q = new ConcurrentLinkedQueue();
469         q.add(one);
470         q.add(two);
471         q.add(three);
472 
473         for (Iterator it = q.iterator(); it.hasNext();) {
474             q.remove();
475             it.next();
476         }
477 
478         assertEquals("queue should be empty again", 0, q.size());
479     }
480 
481     /**
482      * iterator.remove removes current element
483      */
484     public void testIteratorRemove() {
485         final ConcurrentLinkedQueue q = new ConcurrentLinkedQueue();
486         q.add(one);
487         q.add(two);
488         q.add(three);
489         Iterator it = q.iterator();
490         it.next();
491         it.remove();
492         it = q.iterator();
493         assertSame(it.next(), two);
494         assertSame(it.next(), three);
495         assertFalse(it.hasNext());
496     }
497 
498     /**
499      * toString contains toStrings of elements
500      */
501     public void testToString() {
502         ConcurrentLinkedQueue q = populatedQueue(SIZE);
503         String s = q.toString();
504         for (int i = 0; i < SIZE; ++i) {
505             assertTrue(s.contains(String.valueOf(i)));
506         }
507     }
508 
509     /**
510      * A deserialized serialized queue has same elements in same order
511      */
512     public void testSerialization() throws Exception {
513         Queue x = populatedQueue(SIZE);
514         Queue y = serialClone(x);
515 
516         assertNotSame(x, y);
517         assertEquals(x.size(), y.size());
518         assertEquals(x.toString(), y.toString());
519         assertTrue(Arrays.equals(x.toArray(), y.toArray()));
520         while (!x.isEmpty()) {
521             assertFalse(y.isEmpty());
522             assertEquals(x.remove(), y.remove());
523         }
524         assertTrue(y.isEmpty());
525     }
526 
527     /**
528      * remove(null), contains(null) always return false
529      */
530     public void testNeverContainsNull() {
531         Collection<?>[] qs = {
532             new ConcurrentLinkedQueue<Object>(),
533             populatedQueue(2),
534         };
535 
536         for (Collection<?> q : qs) {
537             assertFalse(q.contains(null));
538             assertFalse(q.remove(null));
539         }
540     }
541 }
542