/* * Copyright (C) 2010 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package java.util.concurrent; import java.io.IOException; import java.io.ObjectInputStream; import java.io.ObjectOutputStream; import java.io.Serializable; import java.util.AbstractList; import java.util.Arrays; import java.util.Collection; import java.util.Comparator; import java.util.ConcurrentModificationException; import java.util.Iterator; import java.util.List; import java.util.ListIterator; import java.util.NoSuchElementException; import java.util.RandomAccess; import java.util.function.Consumer; import java.util.function.UnaryOperator; import libcore.util.EmptyArray; import libcore.util.Objects; /** * A thread-safe random-access list. * *

Read operations (including {@link #get}) do not block and may overlap with * update operations. Reads reflect the results of the most recently completed * operations. Aggregate operations like {@link #addAll} and {@link #clear} are * atomic; they never expose an intermediate state. * *

Iterators of this list never throw {@link * ConcurrentModificationException}. When an iterator is created, it keeps a * copy of the list's contents. It is always safe to iterate this list, but * iterations may not reflect the latest state of the list. * *

Iterators returned by this list and its sub lists cannot modify the * underlying list. In particular, {@link Iterator#remove}, {@link * ListIterator#add} and {@link ListIterator#set} all throw {@link * UnsupportedOperationException}. * *

This class offers extended API beyond the {@link List} interface. It * includes additional overloads for indexed search ({@link #indexOf} and {@link * #lastIndexOf}) and methods for conditional adds ({@link #addIfAbsent} and * {@link #addAllAbsent}). */ public class CopyOnWriteArrayList implements List, RandomAccess, Cloneable, Serializable { private static final long serialVersionUID = 8673264195747942595L; /** * Holds the latest snapshot of the list's data. This field is volatile so * that data can be read without synchronization. As a consequence, all * writes to this field must be atomic; it is an error to modify the * contents of an array after it has been assigned to this field. * * Synchronization is required by all update operations. This defends * against one update clobbering the result of another operation. For * example, 100 threads simultaneously calling add() will grow the list's * size by 100 when they have completed. No update operations are lost! * * Maintainers should be careful to read this field only once in * non-blocking read methods. Write methods must be synchronized to avoid * clobbering concurrent writes. */ private transient volatile Object[] elements; /** * Creates a new empty instance. */ public CopyOnWriteArrayList() { elements = EmptyArray.OBJECT; } /** * Creates a new instance containing the elements of {@code collection}. */ @SuppressWarnings("unchecked") public CopyOnWriteArrayList(Collection collection) { this((E[]) collection.toArray()); } /** * Creates a new instance containing the elements of {@code array}. */ public CopyOnWriteArrayList(E[] array) { this.elements = Arrays.copyOf(array, array.length, Object[].class); } @Override public Object clone() { try { CopyOnWriteArrayList result = (CopyOnWriteArrayList) super.clone(); result.elements = result.elements.clone(); return result; } catch (CloneNotSupportedException e) { throw new AssertionError(e); } } public int size() { return elements.length; } @SuppressWarnings("unchecked") public E get(int index) { return (E) elements[index]; } public boolean contains(Object o) { return indexOf(o) != -1; } public boolean containsAll(Collection collection) { Object[] snapshot = elements; return containsAll(collection, snapshot, 0, snapshot.length); } static boolean containsAll(Collection collection, Object[] snapshot, int from, int to) { for (Object o : collection) { if (indexOf(o, snapshot, from, to) == -1) { return false; } } return true; } /** * Searches this list for {@code object} and returns the index of the first * occurrence that is at or after {@code from}. * * @return the index or -1 if the object was not found. */ public int indexOf(E object, int from) { Object[] snapshot = elements; return indexOf(object, snapshot, from, snapshot.length); } public int indexOf(Object object) { Object[] snapshot = elements; return indexOf(object, snapshot, 0, snapshot.length); } /** * Searches this list for {@code object} and returns the index of the last * occurrence that is before {@code to}. * * @return the index or -1 if the object was not found. */ public int lastIndexOf(E object, int to) { Object[] snapshot = elements; return lastIndexOf(object, snapshot, 0, to); } public int lastIndexOf(Object object) { Object[] snapshot = elements; return lastIndexOf(object, snapshot, 0, snapshot.length); } public boolean isEmpty() { return elements.length == 0; } /** * Returns an {@link Iterator} that iterates over the elements of this list * as they were at the time of this method call. Changes to the list made * after this method call will not be reflected by the iterator, nor will * they trigger a {@link ConcurrentModificationException}. * *

The returned iterator does not support {@link Iterator#remove()}. */ public Iterator iterator() { Object[] snapshot = elements; return new CowIterator(snapshot, 0, snapshot.length); } /** * Returns a {@link ListIterator} that iterates over the elements of this * list as they were at the time of this method call. Changes to the list * made after this method call will not be reflected by the iterator, nor * will they trigger a {@link ConcurrentModificationException}. * *

The returned iterator does not support {@link ListIterator#add}, * {@link ListIterator#set} or {@link Iterator#remove()}, */ public ListIterator listIterator(int index) { Object[] snapshot = elements; if (index < 0 || index > snapshot.length) { throw new IndexOutOfBoundsException("index=" + index + ", length=" + snapshot.length); } CowIterator result = new CowIterator(snapshot, 0, snapshot.length); result.index = index; return result; } /** * Equivalent to {@code listIterator(0)}. */ public ListIterator listIterator() { Object[] snapshot = elements; return new CowIterator(snapshot, 0, snapshot.length); } public List subList(int from, int to) { Object[] snapshot = elements; if (from < 0 || from > to || to > snapshot.length) { throw new IndexOutOfBoundsException("from=" + from + ", to=" + to + ", list size=" + snapshot.length); } return new CowSubList(snapshot, from, to); } public Object[] toArray() { return elements.clone(); } @SuppressWarnings({"unchecked","SuspiciousSystemArraycopy"}) public T[] toArray(T[] contents) { Object[] snapshot = elements; if (snapshot.length > contents.length) { return (T[]) Arrays.copyOf(snapshot, snapshot.length, contents.getClass()); } System.arraycopy(snapshot, 0, contents, 0, snapshot.length); if (snapshot.length < contents.length) { contents[snapshot.length] = null; } return contents; } @Override public boolean equals(Object other) { if (other instanceof CopyOnWriteArrayList) { return this == other || Arrays.equals(elements, ((CopyOnWriteArrayList) other).elements); } else if (other instanceof List) { Object[] snapshot = elements; Iterator i = ((List) other).iterator(); for (Object o : snapshot) { if (!i.hasNext() || !Objects.equal(o, i.next())) { return false; } } return !i.hasNext(); } else { return false; } } @Override public int hashCode() { return Arrays.hashCode(elements); } @Override public String toString() { return Arrays.toString(elements); } public synchronized boolean add(E e) { Object[] newElements = new Object[elements.length + 1]; System.arraycopy(elements, 0, newElements, 0, elements.length); newElements[elements.length] = e; elements = newElements; return true; } public synchronized void add(int index, E e) { Object[] newElements = new Object[elements.length + 1]; System.arraycopy(elements, 0, newElements, 0, index); newElements[index] = e; System.arraycopy(elements, index, newElements, index + 1, elements.length - index); elements = newElements; } public synchronized boolean addAll(Collection collection) { return addAll(elements.length, collection); } public synchronized boolean addAll(int index, Collection collection) { Object[] toAdd = collection.toArray(); Object[] newElements = new Object[elements.length + toAdd.length]; System.arraycopy(elements, 0, newElements, 0, index); System.arraycopy(toAdd, 0, newElements, index, toAdd.length); System.arraycopy(elements, index, newElements, index + toAdd.length, elements.length - index); elements = newElements; return toAdd.length > 0; } /** * Adds the elements of {@code collection} that are not already present in * this list. If {@code collection} includes a repeated value, at most one * occurrence of that value will be added to this list. Elements are added * at the end of this list. * *

Callers of this method may prefer {@link CopyOnWriteArraySet}, whose * API is more appropriate for set operations. */ public synchronized int addAllAbsent(Collection collection) { Object[] toAdd = collection.toArray(); Object[] newElements = new Object[elements.length + toAdd.length]; System.arraycopy(elements, 0, newElements, 0, elements.length); int addedCount = 0; for (Object o : toAdd) { if (indexOf(o, newElements, 0, elements.length + addedCount) == -1) { newElements[elements.length + addedCount++] = o; } } if (addedCount < toAdd.length) { newElements = Arrays.copyOfRange( newElements, 0, elements.length + addedCount); // trim to size } elements = newElements; return addedCount; } /** * Adds {@code object} to the end of this list if it is not already present. * *

Callers of this method may prefer {@link CopyOnWriteArraySet}, whose * API is more appropriate for set operations. */ public synchronized boolean addIfAbsent(E object) { if (contains(object)) { return false; } add(object); return true; } @Override public synchronized void clear() { elements = EmptyArray.OBJECT; } public synchronized E remove(int index) { @SuppressWarnings("unchecked") E removed = (E) elements[index]; removeRange(index, index + 1); return removed; } public synchronized boolean remove(Object o) { int index = indexOf(o); if (index == -1) { return false; } remove(index); return true; } public synchronized boolean removeAll(Collection collection) { return removeOrRetain(collection, false, 0, elements.length) != 0; } public synchronized boolean retainAll(Collection collection) { return removeOrRetain(collection, true, 0, elements.length) != 0; } @Override public synchronized void replaceAll(UnaryOperator operator) { replaceInRange(0, elements.length,operator); } private void replaceInRange(int from, int to, UnaryOperator operator) { java.util.Objects.requireNonNull(operator); Object[] newElements = new Object[elements.length]; System.arraycopy(elements, 0, newElements, 0, newElements.length); for (int i = from; i < to; i++) { @SuppressWarnings("unchecked") E e = (E) elements[i]; newElements[i] = operator.apply(e); } elements = newElements; } @Override public synchronized void sort(Comparator c) { sortInRange(0, elements.length, c); } private synchronized void sortInRange(int from, int to, Comparator c) { java.util.Objects.requireNonNull(c); Object[] newElements = new Object[elements.length]; System.arraycopy(elements, 0, newElements, 0, newElements.length); Arrays.sort((E[])newElements, from, to, c); elements = newElements; } @Override public void forEach(Consumer action) { forInRange(0, elements.length, action); } private void forInRange(int from, int to, Consumer action) { java.util.Objects.requireNonNull(action); Object[] newElements = new Object[elements.length]; System.arraycopy(elements, 0, newElements, 0, newElements.length); for (int i = from; i < to; i++) { action.accept((E)newElements[i]); } } /** * Removes or retains the elements in {@code collection}. Returns the number * of elements removed. */ private int removeOrRetain(Collection collection, boolean retain, int from, int to) { for (int i = from; i < to; i++) { if (collection.contains(elements[i]) == retain) { continue; } /* * We've encountered an element that must be removed! Create a new * array and copy in the surviving elements one by one. */ Object[] newElements = new Object[elements.length - 1]; System.arraycopy(elements, 0, newElements, 0, i); int newSize = i; for (int j = i + 1; j < to; j++) { if (collection.contains(elements[j]) == retain) { newElements[newSize++] = elements[j]; } } /* * Copy the elements after 'to'. This is only useful for sub lists, * where 'to' will be less than elements.length. */ System.arraycopy(elements, to, newElements, newSize, elements.length - to); newSize += (elements.length - to); if (newSize < newElements.length) { newElements = Arrays.copyOfRange(newElements, 0, newSize); // trim to size } int removed = elements.length - newElements.length; elements = newElements; return removed; } // we made it all the way through the loop without making any changes return 0; } public synchronized E set(int index, E e) { Object[] newElements = elements.clone(); @SuppressWarnings("unchecked") E result = (E) newElements[index]; newElements[index] = e; elements = newElements; return result; } private void removeRange(int from, int to) { Object[] newElements = new Object[elements.length - (to - from)]; System.arraycopy(elements, 0, newElements, 0, from); System.arraycopy(elements, to, newElements, from, elements.length - to); elements = newElements; } static int lastIndexOf(Object o, Object[] data, int from, int to) { if (o == null) { for (int i = to - 1; i >= from; i--) { if (data[i] == null) { return i; } } } else { for (int i = to - 1; i >= from; i--) { if (o.equals(data[i])) { return i; } } } return -1; } static int indexOf(Object o, Object[] data, int from, int to) { if (o == null) { for (int i = from; i < to; i++) { if (data[i] == null) { return i; } } } else { for (int i = from; i < to; i++) { if (o.equals(data[i])) { return i; } } } return -1; } final Object[] getArray() { // CopyOnWriteArraySet needs this. return elements; } /** * The sub list is thread safe and supports non-blocking reads. Doing so is * more difficult than in the full list, because each read needs to examine * four fields worth of state: * - the elements array of the full list * - two integers for the bounds of this sub list * - the expected elements array (to detect concurrent modification) * * This is accomplished by aggregating the sub list's three fields into a * single snapshot object representing the current slice. This permits reads * to be internally consistent without synchronization. This takes advantage * of Java's concurrency semantics for final fields. */ class CowSubList extends AbstractList { /* * An immutable snapshot of a sub list's state. By gathering all three * of the sub list's fields in an immutable object, */ private volatile Slice slice; public CowSubList(Object[] expectedElements, int from, int to) { this.slice = new Slice(expectedElements, from, to); } @Override public int size() { Slice slice = this.slice; return slice.to - slice.from; } @Override public boolean isEmpty() { Slice slice = this.slice; return slice.from == slice.to; } @SuppressWarnings("unchecked") @Override public E get(int index) { Slice slice = this.slice; Object[] snapshot = elements; slice.checkElementIndex(index); slice.checkConcurrentModification(snapshot); return (E) snapshot[index + slice.from]; } @Override public Iterator iterator() { return listIterator(0); } @Override public ListIterator listIterator() { return listIterator(0); } @Override public ListIterator listIterator(int index) { Slice slice = this.slice; Object[] snapshot = elements; slice.checkPositionIndex(index); slice.checkConcurrentModification(snapshot); CowIterator result = new CowIterator(snapshot, slice.from, slice.to); result.index = slice.from + index; return result; } @Override public int indexOf(Object object) { Slice slice = this.slice; Object[] snapshot = elements; slice.checkConcurrentModification(snapshot); int result = CopyOnWriteArrayList.indexOf(object, snapshot, slice.from, slice.to); return (result != -1) ? (result - slice.from) : -1; } @Override public int lastIndexOf(Object object) { Slice slice = this.slice; Object[] snapshot = elements; slice.checkConcurrentModification(snapshot); int result = CopyOnWriteArrayList.lastIndexOf(object, snapshot, slice.from, slice.to); return (result != -1) ? (result - slice.from) : -1; } @Override public boolean contains(Object object) { return indexOf(object) != -1; } @Override public boolean containsAll(Collection collection) { Slice slice = this.slice; Object[] snapshot = elements; slice.checkConcurrentModification(snapshot); return CopyOnWriteArrayList.containsAll(collection, snapshot, slice.from, slice.to); } @Override public List subList(int from, int to) { Slice slice = this.slice; if (from < 0 || from > to || to > size()) { throw new IndexOutOfBoundsException("from=" + from + ", to=" + to + ", list size=" + size()); } return new CowSubList(slice.expectedElements, slice.from + from, slice.from + to); } @Override public E remove(int index) { synchronized (CopyOnWriteArrayList.this) { slice.checkElementIndex(index); slice.checkConcurrentModification(elements); E removed = CopyOnWriteArrayList.this.remove(slice.from + index); slice = new Slice(elements, slice.from, slice.to - 1); return removed; } } @Override public void clear() { synchronized (CopyOnWriteArrayList.this) { slice.checkConcurrentModification(elements); CopyOnWriteArrayList.this.removeRange(slice.from, slice.to); slice = new Slice(elements, slice.from, slice.from); } } @Override public void add(int index, E object) { synchronized (CopyOnWriteArrayList.this) { slice.checkPositionIndex(index); slice.checkConcurrentModification(elements); CopyOnWriteArrayList.this.add(index + slice.from, object); slice = new Slice(elements, slice.from, slice.to + 1); } } @Override public boolean add(E object) { synchronized (CopyOnWriteArrayList.this) { add(slice.to - slice.from, object); return true; } } @Override public boolean addAll(int index, Collection collection) { synchronized (CopyOnWriteArrayList.this) { slice.checkPositionIndex(index); slice.checkConcurrentModification(elements); int oldSize = elements.length; boolean result = CopyOnWriteArrayList.this.addAll(index + slice.from, collection); slice = new Slice(elements, slice.from, slice.to + (elements.length - oldSize)); return result; } } @Override public boolean addAll(Collection collection) { synchronized (CopyOnWriteArrayList.this) { return addAll(size(), collection); } } @Override public E set(int index, E object) { synchronized (CopyOnWriteArrayList.this) { slice.checkElementIndex(index); slice.checkConcurrentModification(elements); E result = CopyOnWriteArrayList.this.set(index + slice.from, object); slice = new Slice(elements, slice.from, slice.to); return result; } } @Override public boolean remove(Object object) { synchronized (CopyOnWriteArrayList.this) { int index = indexOf(object); if (index == -1) { return false; } remove(index); return true; } } @Override public boolean removeAll(Collection collection) { synchronized (CopyOnWriteArrayList.this) { slice.checkConcurrentModification(elements); int removed = removeOrRetain(collection, false, slice.from, slice.to); slice = new Slice(elements, slice.from, slice.to - removed); return removed != 0; } } @Override public boolean retainAll(Collection collection) { synchronized (CopyOnWriteArrayList.this) { slice.checkConcurrentModification(elements); int removed = removeOrRetain(collection, true, slice.from, slice.to); slice = new Slice(elements, slice.from, slice.to - removed); return removed != 0; } } @Override public void forEach(Consumer action) { CopyOnWriteArrayList.this.forInRange(slice.from, slice.to, action); } @Override public void replaceAll(UnaryOperator operator) { synchronized (CopyOnWriteArrayList.this) { slice.checkConcurrentModification(elements); CopyOnWriteArrayList.this.replaceInRange(slice.from, slice.to, operator); slice = new Slice(elements, slice.from, slice.to); } } @Override public synchronized void sort(Comparator c) { synchronized (CopyOnWriteArrayList.this) { slice.checkConcurrentModification(elements); CopyOnWriteArrayList.this.sortInRange(slice.from, slice.to, c); slice = new Slice(elements, slice.from, slice.to); } } } static class Slice { private final Object[] expectedElements; private final int from; private final int to; Slice(Object[] expectedElements, int from, int to) { this.expectedElements = expectedElements; this.from = from; this.to = to; } /** * Throws if {@code index} doesn't identify an element in the array. */ void checkElementIndex(int index) { if (index < 0 || index >= to - from) { throw new IndexOutOfBoundsException("index=" + index + ", size=" + (to - from)); } } /** * Throws if {@code index} doesn't identify an insertion point in the * array. Unlike element index, it's okay to add or iterate at size(). */ void checkPositionIndex(int index) { if (index < 0 || index > to - from) { throw new IndexOutOfBoundsException("index=" + index + ", size=" + (to - from)); } } void checkConcurrentModification(Object[] snapshot) { if (expectedElements != snapshot) { throw new ConcurrentModificationException(); } } } /** * Iterates an immutable snapshot of the list. */ static class CowIterator implements ListIterator { private final Object[] snapshot; private final int from; private final int to; private int index = 0; CowIterator(Object[] snapshot, int from, int to) { this.snapshot = snapshot; this.from = from; this.to = to; this.index = from; } public void add(E object) { throw new UnsupportedOperationException(); } public boolean hasNext() { return index < to; } public boolean hasPrevious() { return index > from; } @SuppressWarnings("unchecked") public E next() { if (index < to) { return (E) snapshot[index++]; } else { throw new NoSuchElementException(); } } public int nextIndex() { return index; } @SuppressWarnings("unchecked") public E previous() { if (index > from) { return (E) snapshot[--index]; } else { throw new NoSuchElementException(); } } public int previousIndex() { return index - 1; } public void remove() { throw new UnsupportedOperationException(); } public void set(E object) { throw new UnsupportedOperationException(); } @Override public void forEachRemaining(Consumer action) { java.util.Objects.requireNonNull(action); Object[] elements = snapshot; for (int i = index; i < to; i++) { @SuppressWarnings("unchecked") E e = (E) elements[i]; action.accept(e); } index = to; } } private void writeObject(ObjectOutputStream out) throws IOException { Object[] snapshot = elements; out.defaultWriteObject(); out.writeInt(snapshot.length); for (Object o : snapshot) { out.writeObject(o); } } private synchronized void readObject(ObjectInputStream in) throws IOException, ClassNotFoundException { in.defaultReadObject(); Object[] snapshot = new Object[in.readInt()]; for (int i = 0; i < snapshot.length; i++) { snapshot[i] = in.readObject(); } elements = snapshot; } }