2019-05-09 19:32:54 +08:00
|
|
|
package DataStructures.Lists;
|
|
|
|
|
|
|
|
import java.util.Objects;
|
|
|
|
|
2021-10-09 00:38:08 +08:00
|
|
|
/**
|
|
|
|
* This class implements a Cursor Linked List.
|
|
|
|
*
|
|
|
|
* A CursorLinkedList is an array version of a Linked List. Essentially you have an array of list nodes but instead of
|
|
|
|
* each node containing a pointer to the next item in the linked list, each node element in the array contains the index for the next node element.
|
|
|
|
*
|
|
|
|
*/
|
2019-01-30 09:14:57 +08:00
|
|
|
public class CursorLinkedList<T> {
|
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
private static class Node<T> {
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
T element;
|
|
|
|
int next;
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
Node(T element, int next) {
|
|
|
|
this.element = element;
|
|
|
|
this.next = next;
|
2019-01-30 09:14:57 +08:00
|
|
|
}
|
2020-10-24 18:23:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
private final int os;
|
|
|
|
private int head;
|
|
|
|
private final Node<T>[] cursorSpace;
|
|
|
|
private int count;
|
|
|
|
private static final int CURSOR_SPACE_SIZE = 100;
|
|
|
|
|
|
|
|
{
|
|
|
|
// init at loading time
|
|
|
|
cursorSpace = new Node[CURSOR_SPACE_SIZE];
|
|
|
|
for (int i = 0; i < CURSOR_SPACE_SIZE; i++) {
|
|
|
|
cursorSpace[i] = new Node<>(null, i + 1);
|
2019-01-30 09:14:57 +08:00
|
|
|
}
|
2020-10-24 18:23:28 +08:00
|
|
|
cursorSpace[CURSOR_SPACE_SIZE - 1].next = 0;
|
|
|
|
}
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
public CursorLinkedList() {
|
|
|
|
os = 0;
|
|
|
|
count = 0;
|
|
|
|
head = -1;
|
|
|
|
}
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
public void printList() {
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
if (head != -1) {
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
int start = head;
|
|
|
|
while (start != -1) {
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
T element = cursorSpace[start].element;
|
|
|
|
System.out.println(element.toString());
|
|
|
|
start = cursorSpace[start].next;
|
|
|
|
}
|
2019-01-30 09:14:57 +08:00
|
|
|
}
|
2020-10-24 18:23:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return the logical index of the element within the list , not the actual index of the
|
|
|
|
* [cursorSpace] array
|
|
|
|
*/
|
|
|
|
public int indexOf(T element) {
|
|
|
|
|
|
|
|
Objects.requireNonNull(element);
|
|
|
|
Node<T> iterator = cursorSpace[head];
|
|
|
|
for (int i = 0; i < count; i++) {
|
|
|
|
if (iterator.element.equals(element)) {
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
iterator = cursorSpace[iterator.next];
|
2019-01-30 09:14:57 +08:00
|
|
|
}
|
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
return -1;
|
|
|
|
}
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
/**
|
|
|
|
* @param position , the logical index of the element , not the actual one within the
|
|
|
|
* [cursorSpace] array . this method should be used to get the index give by indexOf() method.
|
|
|
|
* @return
|
|
|
|
*/
|
|
|
|
public T get(int position) {
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
if (position >= 0 && position < count) {
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
int start = head;
|
|
|
|
int counter = 0;
|
|
|
|
while (start != -1) {
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
T element = cursorSpace[start].element;
|
|
|
|
if (counter == position) {
|
|
|
|
return element;
|
2019-01-30 09:14:57 +08:00
|
|
|
}
|
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
start = cursorSpace[start].next;
|
|
|
|
counter++;
|
|
|
|
}
|
2019-01-30 09:14:57 +08:00
|
|
|
}
|
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
return null;
|
|
|
|
}
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
public void removeByIndex(int index) {
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
if (index >= 0 && index < count) {
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
T element = get(index);
|
|
|
|
remove(element);
|
2019-01-30 09:14:57 +08:00
|
|
|
}
|
2020-10-24 18:23:28 +08:00
|
|
|
}
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
public void remove(T element) {
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
Objects.requireNonNull(element);
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
// case element is in the head
|
|
|
|
T temp_element = cursorSpace[head].element;
|
|
|
|
int temp_next = cursorSpace[head].next;
|
|
|
|
if (temp_element.equals(element)) {
|
|
|
|
free(head);
|
|
|
|
head = temp_next;
|
|
|
|
} else { // otherwise cases
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
int prev_index = head;
|
|
|
|
int current_index = cursorSpace[prev_index].next;
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
while (current_index != -1) {
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
T current_element = cursorSpace[current_index].element;
|
|
|
|
if (current_element.equals(element)) {
|
|
|
|
cursorSpace[prev_index].next = cursorSpace[current_index].next;
|
|
|
|
free(current_index);
|
|
|
|
break;
|
2019-01-30 09:14:57 +08:00
|
|
|
}
|
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
prev_index = current_index;
|
|
|
|
current_index = cursorSpace[prev_index].next;
|
|
|
|
}
|
2019-01-30 09:14:57 +08:00
|
|
|
}
|
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
count--;
|
|
|
|
}
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
private void free(int index) {
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
Node os_node = cursorSpace[os];
|
|
|
|
int os_next = os_node.next;
|
|
|
|
cursorSpace[os].next = index;
|
|
|
|
cursorSpace[index].element = null;
|
|
|
|
cursorSpace[index].next = os_next;
|
|
|
|
}
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
public void append(T element) {
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
Objects.requireNonNull(element);
|
|
|
|
int availableIndex = alloc();
|
|
|
|
cursorSpace[availableIndex].element = element;
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
if (head == -1) {
|
|
|
|
head = availableIndex;
|
2019-01-30 09:14:57 +08:00
|
|
|
}
|
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
int iterator = head;
|
|
|
|
while (cursorSpace[iterator].next != -1) {
|
|
|
|
iterator = cursorSpace[iterator].next;
|
|
|
|
}
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
cursorSpace[iterator].next = availableIndex;
|
|
|
|
cursorSpace[availableIndex].next = -1;
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
count++;
|
|
|
|
}
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
/** @return the index of the next available node */
|
|
|
|
private int alloc() {
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
// 1- get the index at which the os is pointing
|
|
|
|
int availableNodeIndex = cursorSpace[os].next;
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
if (availableNodeIndex == 0) {
|
|
|
|
throw new OutOfMemoryError();
|
2019-01-30 09:14:57 +08:00
|
|
|
}
|
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
// 2- make the os point to the next of the @var{availableNodeIndex}
|
|
|
|
int availableNext = cursorSpace[availableNodeIndex].next;
|
|
|
|
cursorSpace[os].next = availableNext;
|
|
|
|
|
|
|
|
// this to indicate an end of the list , helpful at testing since any err
|
|
|
|
// would throw an outOfBoundException
|
|
|
|
cursorSpace[availableNodeIndex].next = -1;
|
2019-01-30 09:14:57 +08:00
|
|
|
|
2020-10-24 18:23:28 +08:00
|
|
|
return availableNodeIndex;
|
|
|
|
}
|
2019-01-30 09:14:57 +08:00
|
|
|
}
|