1 /*
2 * ORACLE PROPRIETARY/CONFIDENTIAL. Use is subject to license terms.3 *4 *5 *6 *7 *8 *9 *10 *11 *12 *13 *14 *15 *16 *17 *18 *19 *20 *21 *22 *23 */
24
25 /*
26 *27 *28 *29 *30 *31 * Written by Doug Lea with assistance from members of JCP JSR-16632 * Expert Group and released to the public domain, as explained at33 */publicdomain/zero/1.0/
34 */
35
36 ;37 import .*;38
39 /**
40 * A {@} that uses an internal {@linkCopyOnWriteArrayList}41 * for all of its operations. Thus, it shares the same basic properties:42 *
- 43 *
- It is best suited for applications in which set sizes generally44 * stay small, read-only operations45 * vastly outnumber mutative operations, and you need46 * to prevent interference among threads during traversal.47 *
- It is thread-safe.48 *
- Mutative operations (add, set, remove, etc.)49 * are expensive since they usually entail copying the entire underlying50 * array.51 *
- Iterators do not support the mutative remove operation.52 *
- Traversal via iterators is fast and cannot encounter53 * interference from other threads. Iterators rely on54 * unchanging snapshots of the array at the time the iterators were55 * constructed.56 *
Sample Usage. The following code sketch uses a59 * copy-on-write set to maintain a set of Handler objects that60 * perform some action upon state updates.61 *62 *
{@code
63 * class Handler { void handle(); ... }64 *65 * class X {66 * private final CopyOnWriteArraySet handlers67 * = new CopyOnWriteArraySet();68 * public void addHandler(Handler h) { (h); }69 *70 * private long internalState;71 * private synchronized void changeState() { internalState = ...; }72 *73 * public void update() {74 * changeState();75 * for (Handler handler : handlers)76 * ();77 * }78 * }}
79 *80 *This class is a member of the81 * 82 * Java Collections Framework.83 *84 *@seeCopyOnWriteArrayList85 *@since1.586 *@authorDoug Lea87 *@param the type of elements held in this collection88 */
89 public class CopyOnWriteArraySet extends AbstractSet
90 {91 private static final long serialVersionUID = 5457747651344034263L;92
93 private final CopyOnWriteArrayListal;94
95 /**
96 * Creates an empty set.97 */
98 publicCopyOnWriteArraySet() {99 al = new CopyOnWriteArrayList();100 }101
102 /**
103 * Creates a set containing all of the elements of the specified104 * collection.105 *106 *@paramc the collection of elements to initially contain107 *@throwsNullPointerException if the specified collection is null108 */
109 public CopyOnWriteArraySet(Collection extends E>c) {110 al = new CopyOnWriteArrayList();111 (c);112 }113
114 /**
115 * Returns the number of elements in this set.116 *117 *@returnthe number of elements in this set118 */
119 public intsize() {120 ();121 }122
123 /**
124 * Returns true if this set contains no elements.125 *126 *@returntrue if this set contains no elements127 */
128 public booleanisEmpty() {129 ();130 }131
132 /**
133 * Returns true if this set contains the specified element.134 * More formally, returns true if and only if this set135 * contains an element e such that136 * (o==null ? e==null : (e)).137 *138 *@paramo element whose presence in this set is to be tested139 *@returntrue if this set contains the specified element140 */
141 public booleancontains(Object o) {142 (o);143 }144
145 /**
146 * Returns an array containing all of the elements in this set.147 * If this set makes any guarantees as to what order its elements148 * are returned by its iterator, this method must return the149 * elements in the same order.150 *151 *
The returned array will be "safe" in that no references to it152 * are maintained by this set. (In other words, this method must153 * allocate a new array even if this set is backed by an array).154 * The caller is thus free to modify the returned array.155 *156 *
This method acts as bridge between array-based and collection-based157 * APIs.158 *159 *@returnan array containing all the elements in this set160 */
161 publicObject[] toArray() {162 ();163 }164
165 /**
166 * Returns an array containing all of the elements in this set; the167 * runtime type of the returned array is that of the specified array.168 * If the set fits in the specified array, it is returned therein.169 * Otherwise, a new array is allocated with the runtime type of the170 * specified array and the size of this set.171 *172 *
If this set fits in the specified array with room to spare173 * (., the array has more elements than this set), the element in174 * the array immediately following the end of the set is set to175 * null. (This is useful in determining the length of this176 * set only if the caller knows that this set does not contain177 * any null elements.)178 *179 *
If this set makes any guarantees as to what order its elements180 * are returned by its iterator, this method must return the elements181 * in the same order.182 *183 *
Like the {@link#toArray()} method, this method acts as bridge between184 * array-based and collection-based APIs. Further, this method allows185 * precise control over the runtime type of the output array, and may,186 * under certain circumstances, be used to save allocation costs.187 *188 *
Suppose x is a set known to contain only strings.189 * The following code can be used to dump the set into a newly allocated190 * array of String:191 *192 *
193 * String[] y = (new String[0]);194 *195 * Note that toArray(new Object[0]) is identical in function to196 * toArray().197 *198 *@parama the array into which the elements of this set are to be199 * stored, if it is big enough; otherwise, a new array of the same200 * runtime type is allocated for this purpose.201 *@returnan array containing all the elements in this set202 *@throwsArrayStoreException if the runtime type of the specified array203 * is not a supertype of the runtime type of every element in this204 * set205 *@throwsNullPointerException if the specified array is null206 */
207 public T[] toArray(T[] a) {208 (a);209 }210
211 /**
212 * Removes all of the elements from this set.213 * The set will be empty after this call returns.214 */
215 public voidclear() {216 ();217 }218
219 /**
220 * Removes the specified element from this set if it is present.221 * More formally, removes an element e such that222 * (o==null ? e==null : (e)),223 * if this set contains such an element. Returns true if224 * this set contained the element (or equivalently, if this set225 * changed as a result of the call). (This set will not contain the226 * element once the call returns.)227 *228 *@paramo object to be removed from this set, if present229 *@returntrue if this set contained the specified element230 */
231 public booleanremove(Object o) {232 (o);233 }234
235 /**
236 * Adds the specified element to this set if it is not already present.237 * More formally, adds the specified element e to this set if238 * the set contains no element e2 such that239 * (e==null ? e2==null : (e2)).240 * If this set already contains the element, the call leaves the set241 * unchanged and returns false.242 *243 *@parame element to be added to this set244 *@returntrue if this set did not already contain the specified245 * element246 */
247 public booleanadd(E e) {248 (e);249 }250
251 /**
252 * Returns true if this set contains all of the elements of the253 * specified collection. If the specified collection is also a set, this254 * method returns true if it is a subset of this set.255 *256 *@paramc collection to be checked for containment in this set257 *@returntrue if this set contains all of the elements of the258 * specified collection259 *@throwsNullPointerException if the specified collection is null260 *@see#contains(Object)261 */
262 public boolean containsAll(Collection>c) {263 (c);264 }265
266 /**
267 * Adds all of the elements in the specified collection to this set if268 * they're not already present. If the specified collection is also a269 * set, the addAll operation effectively modifies this set so270 * that its value is the union of the two sets. The behavior of271 * this operation is undefined if the specified collection is modified272 * while the operation is in progress.273 *274 *@paramc collection containing elements to be added to this set275 *@returntrue if this set changed as a result of the call276 *@throwsNullPointerException if the specified collection is null277 *@see#add(Object)278 */
279 public boolean addAll(Collection extends E>c) {280 return (c) > 0;281 }282
283 /**
284 * Removes from this set all of its elements that are contained in the285 * specified collection. If the specified collection is also a set,286 * this operation effectively modifies this set so that its value is the287 * asymmetric set difference of the two sets.288 *289 *@paramc collection containing elements to be removed from this set290 *@returntrue if this set changed as a result of the call291 *@throwsClassCastException if the class of an element of this set292 * is incompatible with the specified collection (optional)293 *@throwsNullPointerException if this set contains a null element and the294 * specified collection does not permit null elements (optional),295 * or if the specified collection is null296 *@see#remove(Object)297 */
298 public boolean removeAll(Collection>c) {299 (c);300 }301
302 /**
303 * Retains only the elements in this set that are contained in the304 * specified collection. In other words, removes from this set all of305 * its elements that are not contained in the specified collection. If306 * the specified collection is also a set, this operation effectively307 * modifies this set so that its value is the intersection of the308 * two sets.309 *310 *@paramc collection containing elements to be retained in this set311 *@returntrue if this set changed as a result of the call312 *@throwsClassCastException if the class of an element of this set313 * is incompatible with the specified collection (optional)314 *@throwsNullPointerException if this set contains a null element and the315 * specified collection does not permit null elements (optional),316 * or if the specified collection is null317 *@see#remove(Object)318 */
319 public boolean retainAll(Collection>c) {320 (c);321 }322
323 /**
324 * Returns an iterator over the elements contained in this set325 * in the order in which these elements were added.326 *327 *
The returned iterator provides a snapshot of the state of the set328 * when the iterator was constructed. No synchronization is needed while329 * traversing the iterator. The iterator does NOT support the330 * remove method.331 *332 *@returnan iterator over the elements in this set333 */
334 public Iteratoriterator() {335 ();336 }337
338 /**
339 * Compares the specified object with this set for equality.340 * Returns {@codetrue} if the specified object is the same object341 * as this object, or if it is also a {@linkSet} and the elements342 * returned by an {@linkplainList#iterator() iterator} over the343 * specified set are the same as the elements returned by an344 * iterator over this set. More formally, the two iterators are345 * considered to return the same elements if they return the same346 * number of elements and for every element {@codee1} returned by347 * the iterator over the specified set, there is an element348 * {@codee2} returned by the iterator over this set such that349 * {@code(e1==null ? e2==null : (e2))}.350 *351 *@paramo object to be compared for equality with this set352 *@return{@codetrue} if the specified object is equal to this set353 */
354 public booleanequals(Object o) {355 if (o == this)356 return true;357 if (!(o instanceofSet))358 return false;359 Set> set = (Set>)(o);360 Iterator> it =();361
362 //Uses O(n^2) algorithm that is only appropriate363 //for small sets, which CopyOnWriteArraySets should be.364
365 //Use a single snapshot of underlying array
366 Object[] elements =();367 int len =;368 //Mark matched elements to avoid re-checking
369 boolean[] matched = new boolean[len];370 int k = 0;371 outer: while(()) {372 if (++k >len)373 return false;374 Object x =();375 for (int i = 0; i < len; ++i) {376 if (!matched[i] &&eq(x, elements[i])) {377 matched[i] = true;378 continueouter;379 }380 }381 return false;382 }383 return k ==len;384 }385
386 /**
387 * Test for equality, coping with nulls.388 */
389 private static booleaneq(Object o1, Object o2) {390 return (o1 == null ? o2 == null: (o2));391 }392 }