org.hsqldb.lib.HsqlDeque Maven / Gradle / Ivy
/* Copyright (c) 2001-2024, 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;
import java.lang.reflect.Array;
import java.util.Arrays;
import java.util.NoSuchElementException;
/**
* A {@code List} that also implements {@code Deque} and {@code Queue}
* and methods for usage as stack.
*
* When used as {@code Queue}, elements are added to the end of
* the List (tail), and retrieved from the start of the List (head).
*
* When used as a stack, elements are added to and retrieved from the start of
* the List (head) using {@code push()} and {@code pop()} methods.
*
* Data is stored in an Object[] that doubles in size when the List gets full
* but does not shrink when it gets empty.
*
* @author Fred Toussi (fredt@users dot sourceforge.net)
* @version 2.7.3
* @since 1.7.0
*/
public class HsqlDeque extends BaseList implements List {
private Object[] list;
private int firstindex = 0; // index of first list element
private int endindex = 0; // index of last list element + 1
// can grow to fill list
// if elementCount == 0 then firstindex == endindex
private static final int DEFAULT_INITIAL_CAPACITY = 16;
public HsqlDeque() {
list = new Object[DEFAULT_INITIAL_CAPACITY];
}
public int size() {
return elementCount;
}
public boolean offer(E e) {
return add(e);
}
public E remove() {
return removeFirst();
}
public E poll() {
if (elementCount == 0) {
return null;
}
return removeFirst();
}
public E element() {
return getFirst();
}
public E peek() {
if (elementCount == 0) {
return null;
}
return getFirst();
}
public void push(E e) {
addFirst(e);
}
public E pop() {
return removeFirst();
}
public E getFirst() throws NoSuchElementException {
if (elementCount == 0) {
throw new NoSuchElementException();
}
return (E) list[firstindex];
}
public E getLast() throws NoSuchElementException {
if (elementCount == 0) {
throw new NoSuchElementException();
}
return (E) list[endindex - 1];
}
public E get(int i) throws IndexOutOfBoundsException {
int index = getInternalIndex(i);
return (E) list[index];
}
public void add(int i, E o) throws IndexOutOfBoundsException {
if (i == elementCount) {
add(o);
return;
}
resetCapacity();
int index = getInternalIndex(i);
if (index < endindex && endindex < list.length) {
System.arraycopy(list, index, list, index + 1, endindex - index);
endindex++;
} else {
System.arraycopy(
list,
firstindex,
list,
firstindex - 1,
index - firstindex);
firstindex--;
index--;
}
list[index] = o;
elementCount++;
}
public E set(int i, E o) throws IndexOutOfBoundsException {
int index = getInternalIndex(i);
E result = (E) list[index];
list[index] = o;
return result;
}
public E removeFirst() throws NoSuchElementException {
if (elementCount == 0) {
throw new NoSuchElementException();
}
E o = (E) list[firstindex];
list[firstindex] = null;
firstindex++;
elementCount--;
if (elementCount == 0) {
firstindex = endindex = 0;
} else if (firstindex == list.length) {
firstindex = 0;
}
return o;
}
public E removeLast() throws NoSuchElementException {
if (elementCount == 0) {
throw new NoSuchElementException();
}
endindex--;
E o = (E) list[endindex];
list[endindex] = null;
elementCount--;
if (elementCount == 0) {
firstindex = endindex = 0;
} else if (endindex == 0) {
endindex = list.length;
}
return o;
}
public E peekFirst() {
return getFirst();
}
public E peekLast() {
return getLast();
}
public boolean offerFirst(E e) {
addFirst(e);
return true;
}
public boolean offerLast(E e) {
addLast(e);
return true;
}
public E pollFirst() {
if (elementCount == 0) {
return null;
}
return removeFirst();
}
public E pollLast() {
if (elementCount == 0) {
return null;
}
return removeLast();
}
public boolean removeFirstOccurrence(E o) {
int index = indexOf(o);
if (index < 1) {
return false;
}
remove(index);
return true;
}
public boolean removeLastOccurrence(E o) {
int index = lastIndexOf(o);
if (index < 1) {
return false;
}
remove(index);
return true;
}
public Iterator descendingIterator() {
throw new UnsupportedOperationException();
}
public boolean add(E o) {
resetCapacity();
if (endindex == list.length) {
endindex = 0;
}
list[endindex] = o;
elementCount++;
endindex++;
return true;
}
public void addLast(E o) {
add(o);
}
public void addFirst(E o) {
resetCapacity();
firstindex--;
if (firstindex < 0) {
firstindex = list.length - 1;
if (endindex == 0) {
endindex = list.length;
}
}
list[firstindex] = o;
elementCount++;
}
public void clear() {
if (elementCount == 0) {
return;
}
firstindex = endindex = elementCount = 0;
Arrays.fill(list, null);
}
public int indexOf(E value) {
for (int i = 0; i < elementCount; i++) {
int index = firstindex + i;
if (index >= list.length) {
index -= list.length;
}
if (list[index] == value) {
return i;
}
if (value != null && comparator.equals(value, (E) list[index])) {
return i;
}
}
return -1;
}
public boolean contains(Object value) {
return indexOf((E) value) >= 0;
}
public boolean remove(E value) {
int i = indexOf(value);
if (i >= 0) {
remove(i);
return true;
}
return false;
}
public int lastIndexOf(E value) {
for (int i = elementCount - 1; i >= 0; i--) {
int index = firstindex + i;
if (index < 0) {
index += list.length;
}
if (list[index] == value) {
return i;
}
if (value != null && comparator.equals(value, (E) list[index])) {
return i;
}
}
return -1;
}
public E remove(int index) {
int target = getInternalIndex(index);
E value = (E) list[target];
if (target == firstindex) {
list[firstindex] = null;
firstindex++;
if (firstindex == list.length) {
firstindex = 0;
}
} else if (target > firstindex) {
System.arraycopy(
list,
firstindex,
list,
firstindex + 1,
target - firstindex);
list[firstindex] = null;
firstindex++;
if (firstindex == list.length) {
firstindex = 0;
}
} else {
System.arraycopy(
list,
target + 1,
list,
target,
endindex - target - 1);
endindex--;
list[endindex] = null;
if (endindex == 0) {
endindex = list.length;
}
}
elementCount--;
if (elementCount == 0) {
firstindex = endindex = 0;
}
return value;
}
private int getInternalIndex(int i) throws IndexOutOfBoundsException {
if (i < 0 || i >= elementCount) {
throw new IndexOutOfBoundsException();
}
int index = firstindex + i;
if (index >= list.length) {
index -= list.length;
}
return index;
}
private void resetCapacity() {
if (elementCount < list.length) {
return;
}
Object[] newList = new Object[list.length * 2];
System.arraycopy(
list,
firstindex,
newList,
firstindex,
list.length - firstindex);
if (endindex <= firstindex) {
System.arraycopy(list, 0, newList, list.length, endindex);
endindex = list.length + endindex;
}
list = newList;
}
public T[] toArray(T[] array) {
if (array.length < elementCount) {
array = (T[]) Array.newInstance(
array.getClass().getComponentType(),
elementCount);
}
int tempCount = list.length - firstindex;
if (tempCount > elementCount) {
tempCount = elementCount;
}
System.arraycopy(list, firstindex, array, 0, tempCount);
if (endindex <= firstindex) {
System.arraycopy(list, 0, array, tempCount, endindex);
}
return array;
}
public Object[] toArray() {
Object[] newArray = new Object[elementCount];
return toArray(newArray);
}
}