| /* Copyright (c) 2001-2010, The HSQL Development Group |
| * All rights reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions are met: |
| * |
| * Redistributions of source code must retain the above copyright notice, this |
| * list of conditions and the following disclaimer. |
| * |
| * Redistributions in binary form must reproduce the above copyright notice, |
| * this list of conditions and the following disclaimer in the documentation |
| * and/or other materials provided with the distribution. |
| * |
| * Neither the name of the HSQL Development Group nor the names of its |
| * contributors may be used to endorse or promote products derived from this |
| * software without specific prior written permission. |
| * |
| * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" |
| * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
| * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE |
| * ARE DISCLAIMED. IN NO EVENT SHALL HSQL DEVELOPMENT GROUP, HSQLDB.ORG, |
| * OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, |
| * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, |
| * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; |
| * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND |
| * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS |
| * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| |
| package org.hsqldb.lib; |
| |
| /** |
| * Intended as an asynchronous alternative to HsqlArrayList. Use HsqlArrayList if |
| * the list won't be initialized sequentially and if frequent references to |
| * specific element positions will be made. |
| * |
| * @author jcpeck@users |
| * @version 1.9.0 |
| * @since 1.7.2 |
| */ |
| public class HsqlLinkedList extends BaseList implements HsqlList { |
| |
| /** |
| * A reference to the head of the list. It is a dummy head (that is, the |
| * Node for index 0 is actually first.next). |
| */ |
| private Node first; |
| |
| /** A reference to the tail of the list */ |
| private Node last; |
| |
| /** |
| * Creates a new instance of HsqlLinkedList. |
| */ |
| public HsqlLinkedList() { |
| |
| first = new Node(null, null); |
| last = first; |
| elementCount = 0; |
| } |
| |
| /** |
| * Returns the first node to allow iterators to be used. |
| */ |
| public Node getHeadNode() { |
| return first; |
| } |
| |
| /** |
| * Removes the given node to allow removel from iterators |
| */ |
| public boolean removeAfter(Node node) { |
| |
| if (node == null || node.next == null) { |
| return false; |
| } |
| |
| if (node.next == last) { |
| last = node; |
| } |
| |
| node.next = node.next.next; |
| |
| return true; |
| } |
| |
| /** |
| * Inserts <code>element</code> at <code>index</code>. |
| * @throws IndexOutOfBoundsException if <code>index</code> < 0 or is > |
| * <code>size</code>. |
| */ |
| public void add(int index, Object element) { |
| |
| if (index == size()) { |
| add(element); //Add to the end of this. |
| } |
| |
| //If index > size() an exception should be thrown with a slightly |
| //different message than the exception thrown by getInternal. |
| else if (index > size()) { |
| throw new IndexOutOfBoundsException("Index out of bounds: " |
| + index + " > " + size()); |
| } else { |
| Node current = getInternal(index); |
| Node newNext = new Node(current.data, current.next); |
| |
| current.data = element; |
| current.next = newNext; |
| |
| elementCount++; |
| |
| //If they inserted at the last valid index, then a new last element |
| //was created, therfore update the last pointer |
| if (last == current) { |
| last = newNext; |
| } |
| } |
| } |
| |
| /** |
| * Appends <code>element</code> to the end of this list. |
| * @return true |
| */ |
| public boolean add(Object element) { |
| |
| last.next = new Node(element, null); |
| last = last.next; |
| |
| elementCount++; |
| |
| return true; |
| } |
| |
| public void clear() { |
| first.next = null; |
| last = first; |
| elementCount = 0; |
| } |
| |
| /** |
| * Gets the element at given position |
| * @throws IndexOutOfBoundsException if index is not valid |
| * index within the list (0 <= <code>index</code> < |
| * <code>size</code>). |
| */ |
| public Object get(int index) { |
| return getInternal(index).data; |
| } |
| |
| /** |
| * Removes and returns the element at <code>index</code>. |
| * @throws IndexOutOfBoundsException if index is not valid |
| * index within the list (0 <= <code>index</code> < |
| * <code>size</code>). |
| */ |
| public Object remove(int index) { |
| |
| //Check that the index is less than size because the getInternal |
| //method is being called with index - 1 and its checks will therefore |
| //not be useful in this case. |
| if (index >= size()) { |
| throw new IndexOutOfBoundsException("Index out of bounds: " |
| + index + " >= " + size()); |
| } |
| |
| //Get the node that is previous to the node being removed |
| Node previousToRemove; |
| |
| if (index == 0) { |
| previousToRemove = first; |
| } else { |
| previousToRemove = getInternal(index - 1); |
| } |
| |
| //previousToRemove.next will never be null because of the check above |
| //that index < size. |
| Node toRemove = previousToRemove.next; |
| |
| previousToRemove.next = toRemove.next; |
| |
| elementCount--; |
| |
| //If they removed at the last valid index, then a the last element |
| //was removed, therfore update the last pointer |
| if (last == toRemove) { |
| last = previousToRemove; |
| } |
| |
| return toRemove.data; |
| } |
| |
| /** |
| * Replaces the current element at <code>index/code> with |
| * <code>element</code>. |
| * @return The current element at <code>index</code>. |
| */ |
| public Object set(int index, Object element) { |
| |
| Node setMe = getInternal(index); |
| Object oldData = setMe.data; |
| |
| setMe.data = element; |
| |
| return oldData; |
| } |
| |
| /** |
| * Accessor for the size of this linked list. The size is the total number |
| * of elements in the list and is one greater than the largest index in the |
| * list. |
| * @return The size of this. |
| */ |
| public final int size() { |
| return elementCount; |
| } |
| |
| /** |
| * Helper method that returns the Node at <code>index</code>. |
| * @param index The index of the Node to return. |
| * @return The Node at the given index. |
| * @throws IndexOutOfBoundsException if index is not valid |
| * index within the list (0 <= <code>index</code> < |
| * <code>size</code>). |
| */ |
| protected final Node getInternal(int index) { |
| |
| //Check preconditions for the index variable |
| if (index >= size()) { |
| throw new IndexOutOfBoundsException("Index out of bounds: " |
| + index + " >= " + size()); |
| } |
| |
| if (index < 0) { |
| throw new IndexOutOfBoundsException("Index out of bounds: " |
| + index + " < 0"); |
| } |
| |
| if (index == 0) { |
| return first.next; |
| } else if (index == (size() - 1)) { |
| return last; |
| } else { |
| Node pointer = first.next; |
| |
| for (int i = 0; i < index; i++) { |
| pointer = pointer.next; |
| } |
| |
| return pointer; |
| } |
| } |
| |
| /** |
| * Inner class that represents nodes within the linked list. This should |
| * be a static inner class to avoid the uneccesary overhead of the |
| * containing class "this" pointer. |
| * jcpeck@users |
| * @version 05/24/2002 |
| */ |
| public static class Node { |
| |
| public Node next; |
| public Object data; |
| |
| public Node() { |
| next = null; |
| data = null; |
| } |
| |
| public Node(Object data) { |
| this.next = null; |
| this.data = data; |
| } |
| |
| public Node(Object data, Node next) { |
| this.next = next; |
| this.data = data; |
| } |
| } |
| } |