All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.infinispan.commons.util.ImmutableListCopy Maven / Gradle / Ivy

There is a newer version: 9.1.7.Final
Show newest version
package org.infinispan.commons.util;

import net.jcip.annotations.Immutable;

import java.io.Externalizable;
import java.io.IOException;
import java.io.ObjectInput;
import java.io.ObjectOutput;
import java.lang.reflect.Array;
import java.util.AbstractList;
import java.util.Collection;
import java.util.Iterator;
import java.util.List;
import java.util.ListIterator;
import java.util.NoSuchElementException;


/**
 * A lightweight, read-only copy of a List.  Typically used in place of the common idiom:  return
 * Collections.unmodifiableList(new ArrayList( myInternalList )); 
 * 

* a it is far more efficient than making a defensive copy and then wrapping the defensive copy in a read-only wrapper. *

* Also used whenever a read-only reference List is needed. *

* * @author Manik Surtani ([email protected]) * @since 4.0 */ @Immutable public class ImmutableListCopy extends AbstractList implements Externalizable, Immutables.Immutable { private static final long serialVersionUID = 10929568968966L; private E[] elements; private int size; /** * Constructs a new ImmutableListCopy. Required by Serialization. */ public ImmutableListCopy() { } /** * Only one copy constructor since the list is immutable. * * @param c collection to copy from */ @SuppressWarnings("unchecked") public ImmutableListCopy(Collection c) { size = c.size(); Object[] el = new Object[size]; // no room for growth; el = c.toArray(el); elements = (E[]) el; } /** * Assumes that the array passed in is "safe", i.e., is not referenced from elsewhere. Use with care! * * @param array to reference */ public ImmutableListCopy(E[] array) { size = array.length; elements = array; } /** * Utility constructors to allow combining collections * * @param collection1 collection to copy from * @param collection2 collection to copy from */ @SuppressWarnings("unchecked") public ImmutableListCopy(Collection collection1, Collection collection2) { size = collection1.size() + collection2.size(); elements = (E[]) new Object[size]; // no room for growth; Object[] c1 = new Object[collection1.size()]; Object[] c2 = new Object[collection2.size()]; c1 = collection1.toArray(c1); c2 = collection2.toArray(c2); System.arraycopy(c1, 0, elements, 0, c1.length); System.arraycopy(c2, 0, elements, c1.length, c2.length); } @Override public final int size() { return size; } @Override public final boolean isEmpty() { return size == 0; } @Override public final boolean contains(Object o) { return indexOf(o) >= 0; } @Override public final Iterator iterator() { return new ImmutableIterator(); } @Override public final Object[] toArray() { Object[] result = new Object[size]; System.arraycopy(elements, 0, result, 0, size); return result; } @Override @SuppressWarnings("unchecked") public final T[] toArray(T[] a) { if (a.length < size) { a = (T[]) Array.newInstance(a.getClass().getComponentType(), size); } System.arraycopy(elements, 0, a, 0, size); if (a.length > size) a[size] = null; return a; } @Override public final boolean add(E o) { throw new UnsupportedOperationException(); } @Override public final boolean remove(Object o) { throw new UnsupportedOperationException(); } @Override public final boolean addAll(Collection c) { throw new UnsupportedOperationException(); } @Override public final boolean addAll(int index, Collection c) { throw new UnsupportedOperationException(); } @Override public final boolean removeAll(Collection c) { throw new UnsupportedOperationException(); } @Override public final boolean retainAll(Collection c) { throw new UnsupportedOperationException(); } @Override public final E get(int index) { if (index >= size || index < 0) throw new IndexOutOfBoundsException("Index: " + index + ", Size: " + size); return elements[index]; } @Override public final int indexOf(Object o) { if (o == null) { for (int i = 0; i < size; i++) { if (elements[i] == null) return i; } } else { for (int i = 0; i < size; i++) { if (o.equals(elements[i])) return i; } } return -1; } @Override public final int lastIndexOf(Object o) { if (o == null) { for (int i = size - 1; i >= 0; i--) { if (elements[i] == null) return i; } } else { for (int i = size - 1; i >= 0; i--) { if (o.equals(elements[i])) return i; } } return -1; } @Override public final ListIterator listIterator() { return new ImmutableIterator(); } @Override public final ListIterator listIterator(int index) { return new ImmutableIterator(index); } @Override public final List subList(int fromIndex, int toIndex) { return new ImmutableSubList(fromIndex, toIndex); } /** * Format: - entry array size (int) - elements (Object) * * @param out stream to write to * @throws IOException */ @Override public void writeExternal(ObjectOutput out) throws IOException { out.writeInt(size); for (E e : elements) out.writeObject(e); } /** * See {@link #writeExternal(java.io.ObjectOutput)} for serialization format * * @param in stream * @throws IOException * @throws ClassNotFoundException */ @Override @SuppressWarnings("unchecked") public void readExternal(ObjectInput in) throws IOException, ClassNotFoundException { size = in.readInt(); elements = (E[]) new Object[size]; for (int i = 0; i < size; i++) elements[i] = (E) in.readObject(); } private class ImmutableIterator implements ListIterator { int cursor = 0; ImmutableIterator(int index) { if (index < 0 || index > size()) throw new IndexOutOfBoundsException("Index: " + index); cursor = index; } ImmutableIterator() { } @Override public boolean hasNext() { return cursor != size; } @Override public E next() { try { return get(cursor++); } catch (IndexOutOfBoundsException e) { throw new NoSuchElementException(); } } @Override public void remove() { throw new UnsupportedOperationException(); } @Override public boolean hasPrevious() { return cursor != 0; } @Override public E previous() { try { return get(--cursor); } catch (IndexOutOfBoundsException e) { throw new NoSuchElementException(); } } @Override public int nextIndex() { return cursor; } @Override public int previousIndex() { return cursor - 1; } @Override public void set(E o) { throw new UnsupportedOperationException(); } @Override public void add(E o) { throw new UnsupportedOperationException(); } } private class ImmutableSubList extends AbstractList { private int offset; private int size; ImmutableSubList(int fromIndex, int toIndex) { if (fromIndex < 0 || toIndex > ImmutableListCopy.this.size || fromIndex > toIndex) throw new IllegalArgumentException("fromIndex(" + fromIndex + "), toIndex(" + toIndex + "), size (" + ImmutableListCopy.this.size + "), List=" + ImmutableListCopy.this.toString()); offset = fromIndex; size = toIndex - fromIndex; } @Override @SuppressWarnings("unchecked") public final E get(int index) { if (index < 0 || index >= size) throw new IndexOutOfBoundsException("Index: " + index + ", Size: " + size); return (E) ImmutableListCopy.this.get(index + offset); } @Override public final int size() { return size; } @Override protected final void removeRange(int fromIndex, int toIndex) { throw new UnsupportedOperationException(); } @Override public final boolean addAll(Collection c) { throw new UnsupportedOperationException(); } @Override public final boolean addAll(int index, Collection c) { throw new UnsupportedOperationException(); } @Override public final Iterator iterator() { return super.listIterator(); } @Override public final ListIterator listIterator(final int index) { if (index < 0 || (index != 0 && index >= size)) throw new IndexOutOfBoundsException("Index: " + index + ", Size: " + size); return new ListIterator() { private ListIterator i = ImmutableListCopy.this.listIterator(index + offset); @Override public boolean hasNext() { return nextIndex() < size; } @Override @SuppressWarnings("unchecked") public E next() { if (hasNext()) return (E) i.next(); else throw new NoSuchElementException(); } @Override public boolean hasPrevious() { return previousIndex() >= 0; } @Override @SuppressWarnings("unchecked") public E previous() { if (hasPrevious()) return (E) i.previous(); else throw new NoSuchElementException(); } @Override public int nextIndex() { return i.nextIndex() - offset; } @Override public int previousIndex() { return i.previousIndex() - offset; } @Override public void remove() { throw new UnsupportedOperationException(); } @Override public void set(E o) { throw new UnsupportedOperationException(); } @Override public void add(E o) { throw new UnsupportedOperationException(); } }; } @Override public final List subList(int fromIndex, int toIndex) { return new ImmutableSubList(offset + fromIndex, offset + toIndex); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy