KoneArrayFixedCapacityLinkedNoddedList
Represents a doubly linked nodded list that is laid out on three arrays of the same fixed capacity instead of using object nodes.
Implementation details
Usual doubly linked list consists of nodes that store elements and references to the next and the previous nodes. In case of this implementation the three jobs are spread between the three arrays data, nextNodeIndex, and previousNodeIndex.
Actually, there are two types of nodes, nodes with elements and nodes without them. All the nodes have their own "actual" indices from 0
to capacity exclusive. They are connected in a (oriented) cycle in a such way that element-containing and empty nodes form two separate connected components. (I.e. at first element-containing nodes are placed in the cycle and then empty nodes are placed in the cycle.) For each node with actual index i
, i
th elements in nextNodeIndex and previousNodeIndex are actual indices of the next and the previous nodes correspondingly. As a corollary,
nextNodeIndex[previousNodeIndex[i]] == i
previousNodeIndex[nextNodeIndex[i]] == i
There are also two references, start and end that are pointing to the first and the last element-containing nodes. But if all the nodes are empty, then start and end are pointing on any two consequent nodes, where end's node is going after start's node.
Then all element-containing nodes are consequently numbered by "virtual" indices from 0
to size exclusive. This virtual indices are exposed as the list's indices.
Finally, the data array contains the i
th actual node in its i
th position. Actual node is an object that is returned by getNode and addNode.
After each operation the structure is preserved in a state that meets the invariants above.
Time complexity of operations
It's obvious that getting actual index corresponding to the list's index takes \(\Theta(\mathrm{size})\) time (both in worst case and in average). Thus, all operations that involve getting actual index take (both in worst case and in average) at least \(\Theta(\mathrm{size})\) time. Other operations that reuse already computed actual indices (including all iterator's and node's operations) take constant time.
Operation | Worst case | Average |
---|---|---|
size | \(\Theta(1)\) | \(\Theta(1)\) |
get | \(\Theta(\mathrm{size})\) | \(\Theta(\mathrm{size})\) |
set | \(\Theta(\mathrm{size})\) | \(\Theta(\mathrm{size})\) |
add | \(\Theta(1)\) | \(\Theta(1)\) |
addAt | \(\Theta(\mathrm{size})\) | \(\Theta(\mathrm{size})\) |
addSeveral | \(\Theta(\mathrm{number})\) | \(\Theta(\mathrm{number})\) |
addSeveralAt | \(\Theta(\mathrm{size} + \mathrm{number})\) | \(\Theta(\mathrm{size} + \mathrm{number})\) |
removeAt | \(\Theta(\mathrm{size})\) | \(\Theta(\mathrm{size})\) |
removeAllThat | \(\Theta(\mathrm{size})\) | \(\Theta(\mathrm{size})\) |
removeAllThatIndexed | \(\Theta(\mathrm{size})\) | \(\Theta(\mathrm{size})\) |
removeAll | \(\Theta(\mathrm{size})\) | \(\Theta(\mathrm{size})\) |
iterator | \(\Theta(1)\) | \(\Theta(1)\) |
iteratorFrom | \(\Theta(\mathrm{size})\) | \(\Theta(\mathrm{size})\) |
KoneSettableLinearIterator.hasNext | \(\Theta(1)\) | \(\Theta(1)\) |
KoneSettableLinearIterator.hasPrevious | \(\Theta(1)\) | \(\Theta(1)\) |
KoneSettableLinearIterator.getNext | \(\Theta(1)\) | \(\Theta(1)\) |
KoneSettableLinearIterator.getPrevious | \(\Theta(1)\) | \(\Theta(1)\) |
KoneSettableLinearIterator.moveNext | \(\Theta(1)\) | \(\Theta(1)\) |
KoneSettableLinearIterator.movePrevious | \(\Theta(1)\) | \(\Theta(1)\) |
KoneSettableLinearIterator.setNext | \(\Theta(1)\) | \(\Theta(1)\) |
KoneSettableLinearIterator.setPrevious | \(\Theta(1)\) | \(\Theta(1)\) |
KoneSettableLinearIterator.setNext | \(\Theta(1)\) | \(\Theta(1)\) |
KoneSettableLinearIterator.setPrevious | \(\Theta(1)\) | \(\Theta(1)\) |
KoneSettableLinearIterator.setNext | \(\Theta(1)\) | \(\Theta(1)\) |
KoneSettableLinearIterator.setPrevious | \(\Theta(1)\) | \(\Theta(1)\) |
KoneSettableLinearIterator.nextIndex | \(\Theta(1)\) | \(\Theta(1)\) |
KoneSettableLinearIterator.previousIndex | \(\Theta(1)\) | \(\Theta(1)\) |
node.element | \(\Theta(1)\) | \(\Theta(1)\) |
node.index | \(\Theta(\mathrm{size})\) | \(\Theta(\mathrm{size})\) |
node.remove | \(\Theta(1)\) | \(\Theta(1)\) |
node.nextNode | \(\Theta(1)\) | \(\Theta(1)\) |
node.previousNode | \(\Theta(1)\) | \(\Theta(1)\) |
node.iteratorFromBeforeHere | \(\Theta(\mathrm{size})\) (for now) | \(\Theta(\mathrm{size})\) (for now) |
node.iteratorFromAfterHere | \(\Theta(\mathrm{size})\) (for now) | \(\Theta(\mathrm{size})\) (for now) |
Functions
Adds provided elements at the end of the ordered collection.
Checks if the iterable is not empty.
Initiates an iterator over the collection's elements with pointer before the first element.
Initiates an iterator over the collection's elements with pointer between elements with indices index - 1
and index
correspondingly.
Removes elements that satisfy the provided predicate.
Removes elements that satisfy the provided predicate.
Iterates over the collection and retains only the elements matching the predicate.
Iterates over the collection and retains only the elements matching the predicate.