001/*
002 * Licensed to the Apache Software Foundation (ASF) under one or more
003 * contributor license agreements.  See the NOTICE file distributed with
004 * this work for additional information regarding copyright ownership.
005 * The ASF licenses this file to You under the Apache License, Version 2.0
006 * (the "License"); you may not use this file except in compliance with
007 * the License.  You may obtain a copy of the License at
008 *
009 *      http://www.apache.org/licenses/LICENSE-2.0
010 *
011 * Unless required by applicable law or agreed to in writing, software
012 * distributed under the License is distributed on an "AS IS" BASIS,
013 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
014 * See the License for the specific language governing permissions and
015 * limitations under the License.
016 */
017package org.apache.commons.collections4.list;
018
019import java.io.IOException;
020import java.io.ObjectInputStream;
021import java.io.ObjectOutputStream;
022import java.lang.reflect.Array;
023import java.util.AbstractList;
024import java.util.Collection;
025import java.util.ConcurrentModificationException;
026import java.util.Iterator;
027import java.util.List;
028import java.util.ListIterator;
029import java.util.NoSuchElementException;
030
031import org.apache.commons.collections4.OrderedIterator;
032
033/**
034 * An abstract implementation of a linked list which provides numerous points for
035 * subclasses to override.
036 * <p>
037 * Overridable methods are provided to change the storage node and to change how
038 * nodes are added to and removed. Hopefully, all you need for unusual subclasses
039 * is here.
040 *
041 * @since 3.0
042 * @version $Id: AbstractLinkedList.LinkedSubList.html 972421 2015-11-14 20:00:04Z tn $
043 */
044public abstract class AbstractLinkedList<E> implements List<E> {
045
046    /*
047     * Implementation notes:
048     * - a standard circular doubly-linked list
049     * - a marker node is stored to mark the start and the end of the list
050     * - node creation and removal always occurs through createNode() and
051     *   removeNode().
052     * - a modification count is kept, with the same semantics as
053     * {@link java.util.LinkedList}.
054     * - respects {@link AbstractList#modCount}
055     */
056
057    /**
058     * A {@link Node} which indicates the start and end of the list and does not
059     * hold a value. The value of <code>next</code> is the first item in the
060     * list. The value of of <code>previous</code> is the last item in the list.
061     */
062    transient Node<E> header;
063
064    /** The size of the list */
065    transient int size;
066
067    /** Modification count for iterators */
068    transient int modCount;
069
070    /**
071     * Constructor that does nothing intended for deserialization.
072     * <p>
073     * If this constructor is used by a serializable subclass then the init()
074     * method must be called.
075     */
076    protected AbstractLinkedList() {
077        super();
078    }
079
080    /**
081     * Constructs a list copying data from the specified collection.
082     *
083     * @param coll  the collection to copy
084     */
085    protected AbstractLinkedList(final Collection<? extends E> coll) {
086        super();
087        init();
088        addAll(coll);
089    }
090
091    /**
092     * The equivalent of a default constructor, broken out so it can be called
093     * by any constructor and by <code>readObject</code>.
094     * Subclasses which override this method should make sure they call super,
095     * so the list is initialised properly.
096     */
097    protected void init() {
098        header = createHeaderNode();
099    }
100
101    //-----------------------------------------------------------------------
102
103    public int size() {
104        return size;
105    }
106
107    public boolean isEmpty() {
108        return size() == 0;
109    }
110
111    public E get(final int index) {
112        final Node<E> node = getNode(index, false);
113        return node.getValue();
114    }
115
116    //-----------------------------------------------------------------------
117
118    public Iterator<E> iterator() {
119        return listIterator();
120    }
121
122    public ListIterator<E> listIterator() {
123        return new LinkedListIterator<E>(this, 0);
124    }
125
126    public ListIterator<E> listIterator(final int fromIndex) {
127        return new LinkedListIterator<E>(this, fromIndex);
128    }
129
130    //-----------------------------------------------------------------------
131
132    public int indexOf(final Object value) {
133        int i = 0;
134        for (Node<E> node = header.next; node != header; node = node.next) {
135            if (isEqualValue(node.getValue(), value)) {
136                return i;
137            }
138            i++;
139        }
140        return -1;
141    }
142
143    public int lastIndexOf(final Object value) {
144        int i = size - 1;
145        for (Node<E> node = header.previous; node != header; node = node.previous) {
146            if (isEqualValue(node.getValue(), value)) {
147                return i;
148            }
149            i--;
150        }
151        return -1;
152    }
153
154    public boolean contains(final Object value) {
155        return indexOf(value) != -1;
156    }
157
158    public boolean containsAll(final Collection<?> coll) {
159        for (final Object o : coll) {
160            if (!contains(o)) {
161                return false;
162            }
163        }
164        return true;
165    }
166
167    //-----------------------------------------------------------------------
168
169    public Object[] toArray() {
170        return toArray(new Object[size]);
171    }
172
173    @SuppressWarnings("unchecked")
174    public <T> T[] toArray(T[] array) {
175        // Extend the array if needed
176        if (array.length < size) {
177            final Class<?> componentType = array.getClass().getComponentType();
178            array = (T[]) Array.newInstance(componentType, size);
179        }
180        // Copy the values into the array
181        int i = 0;
182        for (Node<E> node = header.next; node != header; node = node.next, i++) {
183            array[i] = (T) node.getValue();
184        }
185        // Set the value after the last value to null
186        if (array.length > size) {
187            array[size] = null;
188        }
189        return array;
190    }
191
192    /**
193     * Gets a sublist of the main list.
194     *
195     * @param fromIndexInclusive  the index to start from
196     * @param toIndexExclusive  the index to end at
197     * @return the new sublist
198     */
199    public List<E> subList(final int fromIndexInclusive, final int toIndexExclusive) {
200        return new LinkedSubList<E>(this, fromIndexInclusive, toIndexExclusive);
201    }
202
203    //-----------------------------------------------------------------------
204
205    public boolean add(final E value) {
206        addLast(value);
207        return true;
208    }
209
210    public void add(final int index, final E value) {
211        final Node<E> node = getNode(index, true);
212        addNodeBefore(node, value);
213    }
214
215    public boolean addAll(final Collection<? extends E> coll) {
216        return addAll(size, coll);
217    }
218
219    public boolean addAll(final int index, final Collection<? extends E> coll) {
220        final Node<E> node = getNode(index, true);
221        for (final E e : coll) {
222            addNodeBefore(node, e);
223        }
224        return true;
225    }
226
227    //-----------------------------------------------------------------------
228
229    public E remove(final int index) {
230        final Node<E> node = getNode(index, false);
231        final E oldValue = node.getValue();
232        removeNode(node);
233        return oldValue;
234    }
235
236    public boolean remove(final Object value) {
237        for (Node<E> node = header.next; node != header; node = node.next) {
238            if (isEqualValue(node.getValue(), value)) {
239                removeNode(node);
240                return true;
241            }
242        }
243        return false;
244    }
245
246    /**
247     * {@inheritDoc}
248     * <p>
249     * This implementation iterates over the elements of this list, checking each element in
250     * turn to see if it's contained in <code>coll</code>. If it's contained, it's removed
251     * from this list. As a consequence, it is advised to use a collection type for
252     * <code>coll</code> that provides a fast (e.g. O(1)) implementation of
253     * {@link Collection#contains(Object)}.
254     */
255    public boolean removeAll(final Collection<?> coll) {
256        boolean modified = false;
257        final Iterator<E> it = iterator();
258        while (it.hasNext()) {
259            if (coll.contains(it.next())) {
260                it.remove();
261                modified = true;
262            }
263        }
264        return modified;
265    }
266
267    //-----------------------------------------------------------------------
268
269    /**
270     * {@inheritDoc}
271     * <p>
272     * This implementation iterates over the elements of this list, checking each element in
273     * turn to see if it's contained in <code>coll</code>. If it's not contained, it's removed
274     * from this list. As a consequence, it is advised to use a collection type for
275     * <code>coll</code> that provides a fast (e.g. O(1)) implementation of
276     * {@link Collection#contains(Object)}.
277     */
278    public boolean retainAll(final Collection<?> coll) {
279        boolean modified = false;
280        final Iterator<E> it = iterator();
281        while (it.hasNext()) {
282            if (coll.contains(it.next()) == false) {
283                it.remove();
284                modified = true;
285            }
286        }
287        return modified;
288    }
289
290    public E set(final int index, final E value) {
291        final Node<E> node = getNode(index, false);
292        final E oldValue = node.getValue();
293        updateNode(node, value);
294        return oldValue;
295    }
296
297    public void clear() {
298        removeAllNodes();
299    }
300
301    //-----------------------------------------------------------------------
302
303    public E getFirst() {
304        final Node<E> node = header.next;
305        if (node == header) {
306            throw new NoSuchElementException();
307        }
308        return node.getValue();
309    }
310
311    public E getLast() {
312        final Node<E> node = header.previous;
313        if (node == header) {
314            throw new NoSuchElementException();
315        }
316        return node.getValue();
317    }
318
319    public boolean addFirst(final E o) {
320        addNodeAfter(header, o);
321        return true;
322    }
323
324    public boolean addLast(final E o) {
325        addNodeBefore(header, o);
326        return true;
327    }
328
329    public E removeFirst() {
330        final Node<E> node = header.next;
331        if (node == header) {
332            throw new NoSuchElementException();
333        }
334        final E oldValue = node.getValue();
335        removeNode(node);
336        return oldValue;
337    }
338
339    public E removeLast() {
340        final Node<E> node = header.previous;
341        if (node == header) {
342            throw new NoSuchElementException();
343        }
344        final E oldValue = node.getValue();
345        removeNode(node);
346        return oldValue;
347    }
348
349    //-----------------------------------------------------------------------
350    @Override
351    public boolean equals(final Object obj) {
352        if (obj == this) {
353            return true;
354        }
355        if (obj instanceof List == false) {
356            return false;
357        }
358        final List<?> other = (List<?>) obj;
359        if (other.size() != size()) {
360            return false;
361        }
362        final ListIterator<?> it1 = listIterator();
363        final ListIterator<?> it2 = other.listIterator();
364        while (it1.hasNext() && it2.hasNext()) {
365            final Object o1 = it1.next();
366            final Object o2 = it2.next();
367            if (!(o1 == null ? o2 == null : o1.equals(o2))) {
368                return false;
369            }
370        }
371        return !(it1.hasNext() || it2.hasNext());
372    }
373
374    @Override
375    public int hashCode() {
376        int hashCode = 1;
377        for (final E e : this) {
378            hashCode = 31 * hashCode + (e == null ? 0 : e.hashCode());
379        }
380        return hashCode;
381    }
382
383    @Override
384    public String toString() {
385        if (size() == 0) {
386            return "[]";
387        }
388        final StringBuilder buf = new StringBuilder(16 * size());
389        buf.append('[');
390
391        final Iterator<E> it = iterator();
392        boolean hasNext = it.hasNext();
393        while (hasNext) {
394            final Object value = it.next();
395            buf.append(value == this ? "(this Collection)" : value);
396            hasNext = it.hasNext();
397            if (hasNext) {
398                buf.append(", ");
399            }
400        }
401        buf.append(']');
402        return buf.toString();
403    }
404
405    //-----------------------------------------------------------------------
406    /**
407     * Compares two values for equals.
408     * This implementation uses the equals method.
409     * Subclasses can override this to match differently.
410     *
411     * @param value1  the first value to compare, may be null
412     * @param value2  the second value to compare, may be null
413     * @return true if equal
414     */
415    protected boolean isEqualValue(final Object value1, final Object value2) {
416        return value1 == value2 || (value1 == null ? false : value1.equals(value2));
417    }
418
419    /**
420     * Updates the node with a new value.
421     * This implementation sets the value on the node.
422     * Subclasses can override this to record the change.
423     *
424     * @param node  node to update
425     * @param value  new value of the node
426     */
427    protected void updateNode(final Node<E> node, final E value) {
428        node.setValue(value);
429    }
430
431    /**
432     * Creates a new node with previous, next and element all set to null.
433     * This implementation creates a new empty Node.
434     * Subclasses can override this to create a different class.
435     *
436     * @return  newly created node
437     */
438    protected Node<E> createHeaderNode() {
439        return new Node<E>();
440    }
441
442    /**
443     * Creates a new node with the specified properties.
444     * This implementation creates a new Node with data.
445     * Subclasses can override this to create a different class.
446     *
447     * @param value  value of the new node
448     * @return a new node containing the value
449     */
450    protected Node<E> createNode(final E value) {
451        return new Node<E>(value);
452    }
453
454    /**
455     * Creates a new node with the specified object as its
456     * <code>value</code> and inserts it before <code>node</code>.
457     * <p>
458     * This implementation uses {@link #createNode(Object)} and
459     * {@link #addNode(AbstractLinkedList.Node,AbstractLinkedList.Node)}.
460     *
461     * @param node  node to insert before
462     * @param value  value of the newly added node
463     * @throws NullPointerException if <code>node</code> is null
464     */
465    protected void addNodeBefore(final Node<E> node, final E value) {
466        final Node<E> newNode = createNode(value);
467        addNode(newNode, node);
468    }
469
470    /**
471     * Creates a new node with the specified object as its
472     * <code>value</code> and inserts it after <code>node</code>.
473     * <p>
474     * This implementation uses {@link #createNode(Object)} and
475     * {@link #addNode(AbstractLinkedList.Node,AbstractLinkedList.Node)}.
476     *
477     * @param node  node to insert after
478     * @param value  value of the newly added node
479     * @throws NullPointerException if <code>node</code> is null
480     */
481    protected void addNodeAfter(final Node<E> node, final E value) {
482        final Node<E> newNode = createNode(value);
483        addNode(newNode, node.next);
484    }
485
486    /**
487     * Inserts a new node into the list.
488     *
489     * @param nodeToInsert  new node to insert
490     * @param insertBeforeNode  node to insert before
491     * @throws NullPointerException if either node is null
492     */
493    protected void addNode(final Node<E> nodeToInsert, final Node<E> insertBeforeNode) {
494        nodeToInsert.next = insertBeforeNode;
495        nodeToInsert.previous = insertBeforeNode.previous;
496        insertBeforeNode.previous.next = nodeToInsert;
497        insertBeforeNode.previous = nodeToInsert;
498        size++;
499        modCount++;
500    }
501
502    /**
503     * Removes the specified node from the list.
504     *
505     * @param node  the node to remove
506     * @throws NullPointerException if <code>node</code> is null
507     */
508    protected void removeNode(final Node<E> node) {
509        node.previous.next = node.next;
510        node.next.previous = node.previous;
511        size--;
512        modCount++;
513    }
514
515    /**
516     * Removes all nodes by resetting the circular list marker.
517     */
518    protected void removeAllNodes() {
519        header.next = header;
520        header.previous = header;
521        size = 0;
522        modCount++;
523    }
524
525    /**
526     * Gets the node at a particular index.
527     *
528     * @param index  the index, starting from 0
529     * @param endMarkerAllowed  whether or not the end marker can be returned if
530     * startIndex is set to the list's size
531     * @return the node at the given index
532     * @throws IndexOutOfBoundsException if the index is less than 0; equal to
533     * the size of the list and endMakerAllowed is false; or greater than the
534     * size of the list
535     */
536    protected Node<E> getNode(final int index, final boolean endMarkerAllowed) throws IndexOutOfBoundsException {
537        // Check the index is within the bounds
538        if (index < 0) {
539            throw new IndexOutOfBoundsException("Couldn't get the node: " +
540                    "index (" + index + ") less than zero.");
541        }
542        if (!endMarkerAllowed && index == size) {
543            throw new IndexOutOfBoundsException("Couldn't get the node: " +
544                    "index (" + index + ") is the size of the list.");
545        }
546        if (index > size) {
547            throw new IndexOutOfBoundsException("Couldn't get the node: " +
548                    "index (" + index + ") greater than the size of the " +
549                    "list (" + size + ").");
550        }
551        // Search the list and get the node
552        Node<E> node;
553        if (index < size / 2) {
554            // Search forwards
555            node = header.next;
556            for (int currentIndex = 0; currentIndex < index; currentIndex++) {
557                node = node.next;
558            }
559        } else {
560            // Search backwards
561            node = header;
562            for (int currentIndex = size; currentIndex > index; currentIndex--) {
563                node = node.previous;
564            }
565        }
566        return node;
567    }
568
569    //-----------------------------------------------------------------------
570    /**
571     * Creates an iterator for the sublist.
572     *
573     * @param subList  the sublist to get an iterator for
574     * @return a new iterator on the given sublist
575     */
576    protected Iterator<E> createSubListIterator(final LinkedSubList<E> subList) {
577        return createSubListListIterator(subList, 0);
578    }
579
580    /**
581     * Creates a list iterator for the sublist.
582     *
583     * @param subList  the sublist to get an iterator for
584     * @param fromIndex  the index to start from, relative to the sublist
585     * @return a new list iterator on the given sublist
586     */
587    protected ListIterator<E> createSubListListIterator(final LinkedSubList<E> subList, final int fromIndex) {
588        return new LinkedSubListIterator<E>(subList, fromIndex);
589    }
590
591    //-----------------------------------------------------------------------
592    /**
593     * Serializes the data held in this object to the stream specified.
594     * <p>
595     * The first serializable subclass must call this method from
596     * <code>writeObject</code>.
597     *
598     * @param outputStream  the stream to write the object to
599     * @throws IOException  if anything goes wrong
600     */
601    protected void doWriteObject(final ObjectOutputStream outputStream) throws IOException {
602        // Write the size so we know how many nodes to read back
603        outputStream.writeInt(size());
604        for (final E e : this) {
605            outputStream.writeObject(e);
606        }
607    }
608
609    /**
610     * Deserializes the data held in this object to the stream specified.
611     * <p>
612     * The first serializable subclass must call this method from
613     * <code>readObject</code>.
614     *
615     * @param inputStream  the stream to read the object from
616     * @throws IOException  if any error occurs while reading from the stream
617     * @throws ClassNotFoundException  if a class read from the stream can not be loaded
618     */
619    @SuppressWarnings("unchecked")
620    protected void doReadObject(final ObjectInputStream inputStream) throws IOException, ClassNotFoundException {
621        init();
622        final int size = inputStream.readInt();
623        for (int i = 0; i < size; i++) {
624            add((E) inputStream.readObject());
625        }
626    }
627
628    //-----------------------------------------------------------------------
629    /**
630     * A node within the linked list.
631     * <p>
632     * From Commons Collections 3.1, all access to the <code>value</code> property
633     * is via the methods on this class.
634     */
635    protected static class Node<E> {
636
637        /** A pointer to the node before this node */
638        protected Node<E> previous;
639        /** A pointer to the node after this node */
640        protected Node<E> next;
641        /** The object contained within this node */
642        protected E value;
643
644        /**
645         * Constructs a new header node.
646         */
647        protected Node() {
648            super();
649            previous = this;
650            next = this;
651        }
652
653        /**
654         * Constructs a new node.
655         *
656         * @param value  the value to store
657         */
658        protected Node(final E value) {
659            super();
660            this.value = value;
661        }
662
663        /**
664         * Constructs a new node.
665         *
666         * @param previous  the previous node in the list
667         * @param next  the next node in the list
668         * @param value  the value to store
669         */
670        protected Node(final Node<E> previous, final Node<E> next, final E value) {
671            super();
672            this.previous = previous;
673            this.next = next;
674            this.value = value;
675        }
676
677        /**
678         * Gets the value of the node.
679         *
680         * @return the value
681         * @since 3.1
682         */
683        protected E getValue() {
684            return value;
685        }
686
687        /**
688         * Sets the value of the node.
689         *
690         * @param value  the value
691         * @since 3.1
692         */
693        protected void setValue(final E value) {
694            this.value = value;
695        }
696
697        /**
698         * Gets the previous node.
699         *
700         * @return the previous node
701         * @since 3.1
702         */
703        protected Node<E> getPreviousNode() {
704            return previous;
705        }
706
707        /**
708         * Sets the previous node.
709         *
710         * @param previous  the previous node
711         * @since 3.1
712         */
713        protected void setPreviousNode(final Node<E> previous) {
714            this.previous = previous;
715        }
716
717        /**
718         * Gets the next node.
719         *
720         * @return the next node
721         * @since 3.1
722         */
723        protected Node<E> getNextNode() {
724            return next;
725        }
726
727        /**
728         * Sets the next node.
729         *
730         * @param next  the next node
731         * @since 3.1
732         */
733        protected void setNextNode(final Node<E> next) {
734            this.next = next;
735        }
736    }
737
738    //-----------------------------------------------------------------------
739    /**
740     * A list iterator over the linked list.
741     */
742    protected static class LinkedListIterator<E> implements ListIterator<E>, OrderedIterator<E> {
743
744        /** The parent list */
745        protected final AbstractLinkedList<E> parent;
746
747        /**
748         * The node that will be returned by {@link #next()}. If this is equal
749         * to {@link AbstractLinkedList#header} then there are no more values to return.
750         */
751        protected Node<E> next;
752
753        /**
754         * The index of {@link #next}.
755         */
756        protected int nextIndex;
757
758        /**
759         * The last node that was returned by {@link #next()} or {@link
760         * #previous()}. Set to <code>null</code> if {@link #next()} or {@link
761         * #previous()} haven't been called, or if the node has been removed
762         * with {@link #remove()} or a new node added with {@link #add(Object)}.
763         * Should be accessed through {@link #getLastNodeReturned()} to enforce
764         * this behaviour.
765         */
766        protected Node<E> current;
767
768        /**
769         * The modification count that the list is expected to have. If the list
770         * doesn't have this count, then a
771         * {@link java.util.ConcurrentModificationException} may be thrown by
772         * the operations.
773         */
774        protected int expectedModCount;
775
776        /**
777         * Create a ListIterator for a list.
778         *
779         * @param parent  the parent list
780         * @param fromIndex  the index to start at
781         * @throws IndexOutOfBoundsException if fromIndex is less than 0 or greater than the size of the list
782         */
783        protected LinkedListIterator(final AbstractLinkedList<E> parent, final int fromIndex)
784                throws IndexOutOfBoundsException {
785            super();
786            this.parent = parent;
787            this.expectedModCount = parent.modCount;
788            this.next = parent.getNode(fromIndex, true);
789            this.nextIndex = fromIndex;
790        }
791
792        /**
793         * Checks the modification count of the list is the value that this
794         * object expects.
795         *
796         * @throws ConcurrentModificationException If the list's modification
797         * count isn't the value that was expected.
798         */
799        protected void checkModCount() {
800            if (parent.modCount != expectedModCount) {
801                throw new ConcurrentModificationException();
802            }
803        }
804
805        /**
806         * Gets the last node returned.
807         *
808         * @return the last node returned
809         * @throws IllegalStateException If {@link #next()} or {@link #previous()} haven't been called,
810         * or if the node has been removed with {@link #remove()} or a new node added with {@link #add(Object)}.
811         */
812        protected Node<E> getLastNodeReturned() throws IllegalStateException {
813            if (current == null) {
814                throw new IllegalStateException();
815            }
816            return current;
817        }
818
819        public boolean hasNext() {
820            return next != parent.header;
821        }
822
823        public E next() {
824            checkModCount();
825            if (!hasNext()) {
826                throw new NoSuchElementException("No element at index " + nextIndex + ".");
827            }
828            final E value = next.getValue();
829            current = next;
830            next = next.next;
831            nextIndex++;
832            return value;
833        }
834
835        public boolean hasPrevious() {
836            return next.previous != parent.header;
837        }
838
839        public E previous() {
840            checkModCount();
841            if (!hasPrevious()) {
842                throw new NoSuchElementException("Already at start of list.");
843            }
844            next = next.previous;
845            final E value = next.getValue();
846            current = next;
847            nextIndex--;
848            return value;
849        }
850
851        public int nextIndex() {
852            return nextIndex;
853        }
854
855        public int previousIndex() {
856            // not normally overridden, as relative to nextIndex()
857            return nextIndex() - 1;
858        }
859
860        public void remove() {
861            checkModCount();
862            if (current == next) {
863                // remove() following previous()
864                next = next.next;
865                parent.removeNode(getLastNodeReturned());
866            } else {
867                // remove() following next()
868                parent.removeNode(getLastNodeReturned());
869                nextIndex--;
870            }
871            current = null;
872            expectedModCount++;
873        }
874
875        public void set(final E obj) {
876            checkModCount();
877            getLastNodeReturned().setValue(obj);
878        }
879
880        public void add(final E obj) {
881            checkModCount();
882            parent.addNodeBefore(next, obj);
883            current = null;
884            nextIndex++;
885            expectedModCount++;
886        }
887
888    }
889
890    //-----------------------------------------------------------------------
891    /**
892     * A list iterator over the linked sub list.
893     */
894    protected static class LinkedSubListIterator<E> extends LinkedListIterator<E> {
895
896        /** The parent list */
897        protected final LinkedSubList<E> sub;
898
899        protected LinkedSubListIterator(final LinkedSubList<E> sub, final int startIndex) {
900            super(sub.parent, startIndex + sub.offset);
901            this.sub = sub;
902        }
903
904        @Override
905        public boolean hasNext() {
906            return nextIndex() < sub.size;
907        }
908
909        @Override
910        public boolean hasPrevious() {
911            return previousIndex() >= 0;
912        }
913
914        @Override
915        public int nextIndex() {
916            return super.nextIndex() - sub.offset;
917        }
918
919        @Override
920        public void add(final E obj) {
921            super.add(obj);
922            sub.expectedModCount = parent.modCount;
923            sub.size++;
924        }
925
926        @Override
927        public void remove() {
928            super.remove();
929            sub.expectedModCount = parent.modCount;
930            sub.size--;
931        }
932    }
933
934    //-----------------------------------------------------------------------
935    /**
936     * The sublist implementation for AbstractLinkedList.
937     */
938    protected static class LinkedSubList<E> extends AbstractList<E> {
939        /** The main list */
940        AbstractLinkedList<E> parent;
941        /** Offset from the main list */
942        int offset;
943        /** Sublist size */
944        int size;
945        /** Sublist modCount */
946        int expectedModCount;
947
948        protected LinkedSubList(final AbstractLinkedList<E> parent, final int fromIndex, final int toIndex) {
949            if (fromIndex < 0) {
950                throw new IndexOutOfBoundsException("fromIndex = " + fromIndex);
951            }
952            if (toIndex > parent.size()) {
953                throw new IndexOutOfBoundsException("toIndex = " + toIndex);
954            }
955            if (fromIndex > toIndex) {
956                throw new IllegalArgumentException("fromIndex(" + fromIndex + ") > toIndex(" + toIndex + ")");
957            }
958            this.parent = parent;
959            this.offset = fromIndex;
960            this.size = toIndex - fromIndex;
961            this.expectedModCount = parent.modCount;
962        }
963
964        @Override
965        public int size() {
966            checkModCount();
967            return size;
968        }
969
970        @Override
971        public E get(final int index) {
972            rangeCheck(index, size);
973            checkModCount();
974            return parent.get(index + offset);
975        }
976
977        @Override
978        public void add(final int index, final E obj) {
979            rangeCheck(index, size + 1);
980            checkModCount();
981            parent.add(index + offset, obj);
982            expectedModCount = parent.modCount;
983            size++;
984            LinkedSubList.this.modCount++;
985        }
986
987        @Override
988        public E remove(final int index) {
989            rangeCheck(index, size);
990            checkModCount();
991            final E result = parent.remove(index + offset);
992            expectedModCount = parent.modCount;
993            size--;
994            LinkedSubList.this.modCount++;
995            return result;
996        }
997
998        @Override
999        public boolean addAll(final Collection<? extends E> coll) {
1000            return addAll(size, coll);
1001        }
1002
1003        @Override
1004        public boolean addAll(final int index, final Collection<? extends E> coll) {
1005            rangeCheck(index, size + 1);
1006            final int cSize = coll.size();
1007            if (cSize == 0) {
1008                return false;
1009            }
1010
1011            checkModCount();
1012            parent.addAll(offset + index, coll);
1013            expectedModCount = parent.modCount;
1014            size += cSize;
1015            LinkedSubList.this.modCount++;
1016            return true;
1017        }
1018
1019        @Override
1020        public E set(final int index, final E obj) {
1021            rangeCheck(index, size);
1022            checkModCount();
1023            return parent.set(index + offset, obj);
1024        }
1025
1026        @Override
1027        public void clear() {
1028            checkModCount();
1029            final Iterator<E> it = iterator();
1030            while (it.hasNext()) {
1031                it.next();
1032                it.remove();
1033            }
1034        }
1035
1036        @Override
1037        public Iterator<E> iterator() {
1038            checkModCount();
1039            return parent.createSubListIterator(this);
1040        }
1041
1042        @Override
1043        public ListIterator<E> listIterator(final int index) {
1044            rangeCheck(index, size + 1);
1045            checkModCount();
1046            return parent.createSubListListIterator(this, index);
1047        }
1048
1049        @Override
1050        public List<E> subList(final int fromIndexInclusive, final int toIndexExclusive) {
1051            return new LinkedSubList<E>(parent, fromIndexInclusive + offset, toIndexExclusive + offset);
1052        }
1053
1054        protected void rangeCheck(final int index, final int beyond) {
1055            if (index < 0 || index >= beyond) {
1056                throw new IndexOutOfBoundsException("Index '" + index + "' out of bounds for size '" + size + "'");
1057            }
1058        }
1059
1060        protected void checkModCount() {
1061            if (parent.modCount != expectedModCount) {
1062                throw new ConcurrentModificationException();
1063            }
1064        }
1065    }
1066
1067}