1514 lines
46 KiB
Plaintext
1514 lines
46 KiB
Plaintext
package speiger.src.collections.PACKAGE.maps.impl.customHash;
|
|
|
|
import java.util.Arrays;
|
|
import java.util.Map;
|
|
import java.util.NoSuchElementException;
|
|
import java.util.Objects;
|
|
import java.util.function.Consumer;
|
|
import java.util.function.BiFunction;
|
|
import java.util.function.Predicate;
|
|
#if !TYPE_OBJECT && JDK_TYPE
|
|
import java.util.function.PREDICATE;
|
|
#endif
|
|
#if !SAME_TYPE && JDK_VALUE && !VALUE_OBJECT
|
|
import java.util.function.VALUE_PREDICATE;
|
|
#endif
|
|
|
|
import speiger.src.collections.PACKAGE.collections.BI_ITERATOR;
|
|
#if !TYPE_OBJECT
|
|
import speiger.src.collections.PACKAGE.functions.CONSUMER;
|
|
import speiger.src.collections.objects.functions.consumer.BI_FROM_OBJECT_CONSUMER;
|
|
#endif
|
|
import speiger.src.collections.ints.functions.consumer.BI_FROM_INT_CONSUMER;
|
|
#if !TYPE_OBJECT && !VALUE_OBJECT
|
|
import speiger.src.collections.ints.functions.consumer.IntObjectConsumer;
|
|
#endif
|
|
#if !SAME_TYPE && !TYPE_INT
|
|
import speiger.src.collections.ints.functions.consumer.VALUE_BI_FROM_INT_CONSUMER;
|
|
#endif
|
|
#if !TYPE_OBJECT && !JDK_TYPE
|
|
import speiger.src.collections.PACKAGE.functions.function.PREDICATE;
|
|
#endif
|
|
#if !TYPE_INT || !SAME_TYPE
|
|
import speiger.src.collections.PACKAGE.functions.consumer.BI_CONSUMER;
|
|
#endif
|
|
import speiger.src.collections.PACKAGE.functions.function.SINGLE_UNARY_OPERATOR;
|
|
import speiger.src.collections.PACKAGE.lists.LIST_ITERATOR;
|
|
import speiger.src.collections.PACKAGE.maps.interfaces.MAP;
|
|
import speiger.src.collections.PACKAGE.maps.interfaces.ORDERED_MAP;
|
|
import speiger.src.collections.PACKAGE.sets.ABSTRACT_SET;
|
|
import speiger.src.collections.PACKAGE.sets.ORDERED_SET;
|
|
import speiger.src.collections.PACKAGE.utils.STRATEGY;
|
|
import speiger.src.collections.VALUE_PACKAGE.collections.VALUE_ABSTRACT_COLLECTION;
|
|
import speiger.src.collections.VALUE_PACKAGE.collections.VALUE_COLLECTION;
|
|
import speiger.src.collections.VALUE_PACKAGE.collections.VALUE_ITERATOR;
|
|
#if !SAME_TYPE
|
|
import speiger.src.collections.VALUE_PACKAGE.functions.function.VALUE_UNARY_OPERATOR;
|
|
#endif
|
|
#if !VALUE_OBJECT && !SAME_TYPE
|
|
import speiger.src.collections.VALUE_PACKAGE.functions.VALUE_CONSUMER;
|
|
import speiger.src.collections.VALUE_PACKAGE.lists.VALUE_LIST_ITERATOR;
|
|
#endif
|
|
#if !TYPE_OBJECT && !VALUE_OBJECT || !VALUE_OBJECT
|
|
import speiger.src.collections.objects.functions.consumer.ObjectObjectConsumer;
|
|
#endif
|
|
#if !SAME_TYPE
|
|
#if !TYPE_OBJECT
|
|
import speiger.src.collections.objects.functions.consumer.VALUE_BI_FROM_OBJECT_CONSUMER;
|
|
|
|
#endif
|
|
#if !JDK_VALUE
|
|
import speiger.src.collections.VALUE_PACKAGE.functions.function.VALUE_PREDICATE;
|
|
#endif
|
|
#endif
|
|
#if !TYPE_OBJECT
|
|
#if !VALUE_OBJECT
|
|
import speiger.src.collections.objects.functions.function.ObjectObjectUnaryOperator;
|
|
|
|
#endif
|
|
import speiger.src.collections.objects.collections.ObjectBidirectionalIterator;
|
|
import speiger.src.collections.objects.lists.ObjectListIterator;
|
|
import speiger.src.collections.objects.sets.AbstractObjectSet;
|
|
import speiger.src.collections.objects.sets.ObjectOrderedSet;
|
|
#endif
|
|
import speiger.src.collections.utils.HashUtil;
|
|
|
|
/**
|
|
* A Type Specific LinkedHashMap that allows for custom HashControl. That uses arrays to create links between nodes.
|
|
* For cases where Objects/primitive do not allow hashcoding this can be really useful and provide a lot of control.
|
|
* This implementation of SortedMap does not support SubMaps of any kind. It implements the interface due to sortability and first/last access
|
|
* @Type(T)
|
|
* @ValueType(V)
|
|
*/
|
|
public class LINKED_CUSTOM_HASH_MAP KEY_VALUE_GENERIC_TYPE extends CUSTOM_HASH_MAP KEY_VALUE_GENERIC_TYPE implements ORDERED_MAP KEY_VALUE_GENERIC_TYPE
|
|
{
|
|
/** The Backing array for links between nodes. Left 32 Bits => Previous Entry, Right 32 Bits => Next Entry */
|
|
protected transient long[] links;
|
|
/** The First Index in the Map */
|
|
protected int firstIndex = -1;
|
|
/** The Last Index in the Map */
|
|
protected int lastIndex = -1;
|
|
|
|
/**
|
|
* Default Constructor
|
|
* @param strategy the strategy that allows hash control.
|
|
* @throws NullPointerException if Strategy is null
|
|
*/
|
|
public LINKED_CUSTOM_HASH_MAP(STRATEGY KEY_SUPER_GENERIC_TYPE strategy) {
|
|
this(HashUtil.DEFAULT_MIN_CAPACITY, HashUtil.DEFAULT_LOAD_FACTOR, strategy);
|
|
}
|
|
|
|
/**
|
|
* Constructor that defines the minimum capacity
|
|
* @param minCapacity the minimum capacity the HashMap is allowed to be.
|
|
* @param strategy the strategy that allows hash control.
|
|
* @throws NullPointerException if Strategy is null
|
|
* @throws IllegalStateException if the minimum capacity is negative
|
|
*/
|
|
public LINKED_CUSTOM_HASH_MAP(int minCapacity, STRATEGY KEY_SUPER_GENERIC_TYPE strategy) {
|
|
this(minCapacity, HashUtil.DEFAULT_LOAD_FACTOR, strategy);
|
|
}
|
|
|
|
/**
|
|
* Constructor that defines the minimum capacity and load factor
|
|
* @param minCapacity the minimum capacity the HashMap is allowed to be.
|
|
* @param loadFactor the percentage of how full the backing array can be before they resize
|
|
* @param strategy the strategy that allows hash control.
|
|
* @throws NullPointerException if Strategy is null
|
|
* @throws IllegalStateException if the minimum capacity is negative
|
|
* @throws IllegalStateException if the loadfactor is either below/equal to 0 or above/equal to 1
|
|
*/
|
|
public LINKED_CUSTOM_HASH_MAP(int minCapacity, float loadFactor, STRATEGY KEY_SUPER_GENERIC_TYPE strategy) {
|
|
super(minCapacity, loadFactor, strategy);
|
|
links = new long[nullIndex + 1];
|
|
}
|
|
|
|
#if !TYPE_OBJECT || !VALUE_OBJECT
|
|
/**
|
|
* Helper constructor that allow to create a map from boxed values (it will unbox them)
|
|
* @param keys the keys that should be put into the map
|
|
* @param values the values that should be put into the map.
|
|
* @param strategy the strategy that allows hash control.
|
|
* @throws NullPointerException if Strategy is null
|
|
* @throws IllegalStateException if the keys and values do not match in lenght
|
|
*/
|
|
public LINKED_CUSTOM_HASH_MAP(CLASS_TYPE[] keys, CLASS_VALUE_TYPE[] values, STRATEGY KEY_SUPER_GENERIC_TYPE strategy) {
|
|
this(keys, values, HashUtil.DEFAULT_LOAD_FACTOR, strategy);
|
|
}
|
|
|
|
/**
|
|
* Helper constructor that allow to create a map from boxed values (it will unbox them)
|
|
* @param keys the keys that should be put into the map
|
|
* @param values the values that should be put into the map.
|
|
* @param loadFactor the percentage of how full the backing array can be before they resize
|
|
* @param strategy the strategy that allows hash control.
|
|
* @throws NullPointerException if Strategy is null
|
|
* @throws IllegalStateException if the keys and values do not match in lenght
|
|
* @throws IllegalStateException if the loadfactor is either below/equal to 0 or above/equal to 1
|
|
*/
|
|
public LINKED_CUSTOM_HASH_MAP(CLASS_TYPE[] keys, CLASS_VALUE_TYPE[] values, float loadFactor, STRATEGY KEY_SUPER_GENERIC_TYPE strategy) {
|
|
this(keys.length, loadFactor, strategy);
|
|
if(keys.length != values.length) throw new IllegalStateException("Input Arrays are not equal size");
|
|
for(int i = 0,m=keys.length;i<m;i++) put(OBJ_TO_KEY(keys[i]), OBJ_TO_VALUE(values[i]));
|
|
}
|
|
|
|
#endif
|
|
/**
|
|
* Helper constructor that allow to create a map from unboxed values
|
|
* @param keys the keys that should be put into the map
|
|
* @param values the values that should be put into the map.
|
|
* @param strategy the strategy that allows hash control.
|
|
* @throws NullPointerException if Strategy is null
|
|
* @throws IllegalStateException if the keys and values do not match in lenght
|
|
*/
|
|
public LINKED_CUSTOM_HASH_MAP(KEY_TYPE[] keys, VALUE_TYPE[] values, STRATEGY KEY_SUPER_GENERIC_TYPE strategy) {
|
|
this(keys, values, HashUtil.DEFAULT_LOAD_FACTOR, strategy);
|
|
}
|
|
|
|
/**
|
|
* Helper constructor that allow to create a map from unboxed values
|
|
* @param keys the keys that should be put into the map
|
|
* @param values the values that should be put into the map.
|
|
* @param loadFactor the percentage of how full the backing array can be before they resize
|
|
* @param strategy the strategy that allows hash control.
|
|
* @throws NullPointerException if Strategy is null
|
|
* @throws IllegalStateException if the keys and values do not match in lenght
|
|
* @throws IllegalStateException if the loadfactor is either below/equal to 0 or above/equal to 1
|
|
*/
|
|
public LINKED_CUSTOM_HASH_MAP(KEY_TYPE[] keys, VALUE_TYPE[] values, float loadFactor, STRATEGY KEY_SUPER_GENERIC_TYPE strategy) {
|
|
this(keys.length, loadFactor, strategy);
|
|
if(keys.length != values.length) throw new IllegalStateException("Input Arrays are not equal size");
|
|
for(int i = 0,m=keys.length;i<m;i++) put(keys[i], values[i]);
|
|
}
|
|
|
|
/**
|
|
* A Helper constructor that allows to create a Map with exactly the same values as the provided map.
|
|
* @param map the values that should be present in the map
|
|
* @param strategy the strategy that allows hash control.
|
|
* @throws NullPointerException if Strategy is null
|
|
*/
|
|
public LINKED_CUSTOM_HASH_MAP(Map<? extends CLASS_TYPE, ? extends CLASS_VALUE_TYPE> map, STRATEGY KEY_SUPER_GENERIC_TYPE strategy) {
|
|
this(map, HashUtil.DEFAULT_LOAD_FACTOR, strategy);
|
|
}
|
|
|
|
/**
|
|
* A Helper constructor that allows to create a Map with exactly the same values as the provided map.
|
|
* @param map the values that should be present in the map
|
|
* @param loadFactor the percentage of how full the backing array can be before they resize
|
|
* @param strategy the strategy that allows hash control.
|
|
* @throws NullPointerException if Strategy is null
|
|
* @throws IllegalStateException if the loadfactor is either below/equal to 0 or above/equal to 1
|
|
*/
|
|
public LINKED_CUSTOM_HASH_MAP(Map<? extends CLASS_TYPE, ? extends CLASS_VALUE_TYPE> map, float loadFactor, STRATEGY KEY_SUPER_GENERIC_TYPE strategy) {
|
|
this(map.size(), loadFactor, strategy);
|
|
putAll(map);
|
|
}
|
|
|
|
/**
|
|
* A Type Specific Helper function that allows to create a new Map with exactly the same values as the provided map.
|
|
* @param map the values that should be present in the map
|
|
* @param strategy the strategy that allows hash control.
|
|
* @throws NullPointerException if Strategy is null
|
|
*/
|
|
public LINKED_CUSTOM_HASH_MAP(MAP KEY_VALUE_GENERIC_TYPE map, STRATEGY KEY_SUPER_GENERIC_TYPE strategy) {
|
|
this(map, HashUtil.DEFAULT_LOAD_FACTOR, strategy);
|
|
}
|
|
|
|
/**
|
|
* A Type Specific Helper function that allows to create a new Map with exactly the same values as the provided map.
|
|
* @param map the values that should be present in the map
|
|
* @param loadFactor the percentage of how full the backing array can be before they resize
|
|
* @param strategy the strategy that allows hash control.
|
|
* @throws NullPointerException if Strategy is null
|
|
* @throws IllegalStateException if the loadfactor is either below/equal to 0 or above/equal to 1
|
|
*/
|
|
public LINKED_CUSTOM_HASH_MAP(MAP KEY_VALUE_GENERIC_TYPE map, float loadFactor, STRATEGY KEY_SUPER_GENERIC_TYPE strategy) {
|
|
this(map.size(), loadFactor, strategy);
|
|
putAll(map);
|
|
}
|
|
|
|
@Override
|
|
public VALUE_TYPE putAndMoveToFirst(KEY_TYPE key, VALUE_TYPE value) {
|
|
if(strategy.equals(key, EMPTY_KEY_VALUE)) {
|
|
if(containsNull) {
|
|
VALUE_TYPE lastValue = values[nullIndex];
|
|
values[nullIndex] = value;
|
|
moveToFirstIndex(nullIndex);
|
|
return lastValue;
|
|
}
|
|
values[nullIndex] = value;
|
|
containsNull = true;
|
|
onNodeAdded(nullIndex);
|
|
moveToFirstIndex(nullIndex);
|
|
}
|
|
else {
|
|
int pos = HashUtil.mix(strategy.hashCode(key)) & mask;
|
|
while(!strategy.equals(keys[pos], EMPTY_KEY_VALUE)) {
|
|
if(strategy.equals(keys[pos], key)) {
|
|
VALUE_TYPE lastValue = values[pos];
|
|
values[pos] = value;
|
|
moveToFirstIndex(pos);
|
|
return lastValue;
|
|
}
|
|
pos = ++pos & mask;
|
|
}
|
|
keys[pos] = key;
|
|
values[pos] = value;
|
|
onNodeAdded(pos);
|
|
moveToFirstIndex(pos);
|
|
}
|
|
if(size++ >= maxFill) rehash(HashUtil.arraySize(size+1, loadFactor));
|
|
return getDefaultReturnValue();
|
|
}
|
|
|
|
@Override
|
|
public VALUE_TYPE putAndMoveToLast(KEY_TYPE key, VALUE_TYPE value) {
|
|
if(strategy.equals(key, EMPTY_KEY_VALUE)) {
|
|
if(containsNull) {
|
|
VALUE_TYPE lastValue = values[nullIndex];
|
|
values[nullIndex] = value;
|
|
moveToLastIndex(nullIndex);
|
|
return lastValue;
|
|
}
|
|
values[nullIndex] = value;
|
|
containsNull = true;
|
|
onNodeAdded(nullIndex);
|
|
moveToLastIndex(nullIndex);
|
|
}
|
|
else {
|
|
int pos = HashUtil.mix(strategy.hashCode(key)) & mask;
|
|
while(!strategy.equals(keys[pos], EMPTY_KEY_VALUE)) {
|
|
if(strategy.equals(keys[pos], key)) {
|
|
VALUE_TYPE lastValue = values[pos];
|
|
values[pos] = value;
|
|
moveToLastIndex(pos);
|
|
return lastValue;
|
|
}
|
|
pos = ++pos & mask;
|
|
}
|
|
keys[pos] = key;
|
|
values[pos] = value;
|
|
onNodeAdded(pos);
|
|
moveToLastIndex(pos);
|
|
}
|
|
if(size++ >= maxFill) rehash(HashUtil.arraySize(size+1, loadFactor));
|
|
return getDefaultReturnValue();
|
|
}
|
|
|
|
@Override
|
|
public boolean moveToFirst(KEY_TYPE key) {
|
|
if(isEmpty() || strategy.equals(FIRST_ENTRY_KEY(), key)) return false;
|
|
if(strategy.equals(key, EMPTY_KEY_VALUE)) {
|
|
if(containsNull) {
|
|
moveToFirstIndex(nullIndex);
|
|
return true;
|
|
}
|
|
}
|
|
else {
|
|
int pos = HashUtil.mix(strategy.hashCode(key)) & mask;
|
|
while(!strategy.equals(keys[pos], EMPTY_KEY_VALUE)) {
|
|
if(strategy.equals(keys[pos], key)) {
|
|
moveToFirstIndex(pos);
|
|
return true;
|
|
}
|
|
pos = ++pos & mask;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
@Override
|
|
public boolean moveToLast(KEY_TYPE key) {
|
|
if(isEmpty() || strategy.equals(LAST_ENTRY_KEY(), key)) return false;
|
|
if(strategy.equals(key, EMPTY_KEY_VALUE)) {
|
|
if(containsNull) {
|
|
moveToLastIndex(nullIndex);
|
|
return true;
|
|
}
|
|
}
|
|
else {
|
|
int pos = HashUtil.mix(strategy.hashCode(key)) & mask;
|
|
while(!strategy.equals(keys[pos], EMPTY_KEY_VALUE)) {
|
|
if(strategy.equals(keys[pos], key)) {
|
|
moveToLastIndex(pos);
|
|
return true;
|
|
}
|
|
pos = ++pos & mask;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
@Override
|
|
public VALUE_TYPE getAndMoveToFirst(KEY_TYPE key) {
|
|
int index = findIndex(key);
|
|
if(index < 0) return getDefaultReturnValue();
|
|
moveToFirstIndex(index);
|
|
return values[index];
|
|
}
|
|
|
|
@Override
|
|
public VALUE_TYPE getAndMoveToLast(KEY_TYPE key) {
|
|
int index = findIndex(key);
|
|
if(index < 0) return getDefaultReturnValue();
|
|
moveToLastIndex(index);
|
|
return values[index];
|
|
}
|
|
|
|
@Override
|
|
public LINKED_CUSTOM_HASH_MAP KEY_VALUE_GENERIC_TYPE copy() {
|
|
LINKED_CUSTOM_HASH_MAP KEY_VALUE_GENERIC_TYPE map = new LINKED_CUSTOM_HASH_MAPKV_BRACES(0, loadFactor, strategy);
|
|
map.minCapacity = minCapacity;
|
|
map.mask = mask;
|
|
map.maxFill = maxFill;
|
|
map.nullIndex = nullIndex;
|
|
map.containsNull = containsNull;
|
|
map.size = size;
|
|
map.keys = Arrays.copyOf(keys, keys.length);
|
|
map.values = Arrays.copyOf(values, values.length);
|
|
map.links = Arrays.copyOf(links, links.length);
|
|
map.firstIndex = firstIndex;
|
|
map.lastIndex = lastIndex;
|
|
return map;
|
|
}
|
|
|
|
@Override
|
|
public KEY_TYPE FIRST_ENTRY_KEY() {
|
|
if(size == 0) throw new NoSuchElementException();
|
|
return keys[firstIndex];
|
|
}
|
|
|
|
@Override
|
|
public KEY_TYPE POLL_FIRST_ENTRY_KEY() {
|
|
if(size == 0) throw new NoSuchElementException();
|
|
int pos = firstIndex;
|
|
onNodeRemoved(pos);
|
|
KEY_TYPE result = keys[pos];
|
|
size--;
|
|
if(strategy.equals(result, EMPTY_KEY_VALUE)) {
|
|
containsNull = false;
|
|
keys[nullIndex] = EMPTY_KEY_VALUE;
|
|
values[nullIndex] = EMPTY_VALUE;
|
|
}
|
|
else shiftKeys(pos);
|
|
if(nullIndex > minCapacity && size < maxFill / 4 && nullIndex > HashUtil.DEFAULT_MIN_CAPACITY) rehash(nullIndex / 2);
|
|
return result;
|
|
}
|
|
|
|
@Override
|
|
public KEY_TYPE LAST_ENTRY_KEY() {
|
|
if(size == 0) throw new NoSuchElementException();
|
|
return keys[lastIndex];
|
|
}
|
|
|
|
@Override
|
|
public KEY_TYPE POLL_LAST_ENTRY_KEY() {
|
|
if(size == 0) throw new NoSuchElementException();
|
|
int pos = lastIndex;
|
|
onNodeRemoved(pos);
|
|
KEY_TYPE result = keys[pos];
|
|
size--;
|
|
if(strategy.equals(result, EMPTY_KEY_VALUE)) {
|
|
containsNull = false;
|
|
keys[nullIndex] = EMPTY_KEY_VALUE;
|
|
values[nullIndex] = EMPTY_VALUE;
|
|
}
|
|
else shiftKeys(pos);
|
|
if(nullIndex > minCapacity && size < maxFill / 4 && nullIndex > HashUtil.DEFAULT_MIN_CAPACITY) rehash(nullIndex / 2);
|
|
return result;
|
|
}
|
|
|
|
@Override
|
|
public VALUE_TYPE FIRST_ENTRY_VALUE() {
|
|
if(size == 0) throw new NoSuchElementException();
|
|
return values[firstIndex];
|
|
}
|
|
|
|
@Override
|
|
public VALUE_TYPE LAST_ENTRY_VALUE() {
|
|
if(size == 0) throw new NoSuchElementException();
|
|
return values[lastIndex];
|
|
}
|
|
|
|
@Override
|
|
public ObjectOrderedSet<MAP.Entry KEY_VALUE_GENERIC_TYPE> ENTRY_SET() {
|
|
if(entrySet == null) entrySet = new MapEntrySet();
|
|
return (ObjectOrderedSet<MAP.Entry KEY_VALUE_GENERIC_TYPE>)entrySet;
|
|
}
|
|
|
|
@Override
|
|
public ORDERED_SET KEY_GENERIC_TYPE keySet() {
|
|
if(keySet == null) keySet = new KeySet();
|
|
return (ORDERED_SET KEY_GENERIC_TYPE)keySet;
|
|
}
|
|
|
|
@Override
|
|
public VALUE_COLLECTION VALUE_GENERIC_TYPE values() {
|
|
if(valuesC == null) valuesC = new Values();
|
|
return valuesC;
|
|
}
|
|
|
|
@Override
|
|
public void forEach(BI_CONSUMER KEY_VALUE_GENERIC_TYPE action) {
|
|
int index = firstIndex;
|
|
while(index != -1){
|
|
action.accept(keys[index], values[index]);
|
|
index = (int)links[index];
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void clear() {
|
|
super.clear();
|
|
firstIndex = lastIndex = -1;
|
|
}
|
|
|
|
@Override
|
|
public void clearAndTrim(int size) {
|
|
int request = Math.max(minCapacity, HashUtil.nextPowerOfTwo((int)Math.ceil(size / loadFactor)));
|
|
if(request >= nullIndex) {
|
|
clear();
|
|
return;
|
|
}
|
|
nullIndex = request;
|
|
mask = request-1;
|
|
maxFill = Math.min((int)Math.ceil(nullIndex * loadFactor), nullIndex - 1);
|
|
keys = NEW_KEY_ARRAY(request + 1);
|
|
values = NEW_VALUE_ARRAY(request + 1);
|
|
links = new long[request + 1];
|
|
firstIndex = lastIndex = -1;
|
|
this.size = 0;
|
|
containsNull = false;
|
|
}
|
|
|
|
protected void moveToFirstIndex(int startPos) {
|
|
if(size == 1 || firstIndex == startPos) return;
|
|
if(lastIndex == startPos) {
|
|
lastIndex = (int)(links[startPos] >>> 32);
|
|
links[lastIndex] |= 0xFFFFFFFFL;
|
|
}
|
|
else {
|
|
long link = links[startPos];
|
|
int prev = (int)(link >>> 32);
|
|
int next = (int)link;
|
|
links[prev] ^= ((links[prev] ^ (link & 0xFFFFFFFFL)) & 0xFFFFFFFFL);
|
|
links[next] ^= ((links[next] ^ (link & 0xFFFFFFFF00000000L)) & 0xFFFFFFFF00000000L);
|
|
}
|
|
links[firstIndex] ^= ((links[firstIndex] ^ ((startPos & 0xFFFFFFFFL) << 32)) & 0xFFFFFFFF00000000L);
|
|
links[startPos] = 0xFFFFFFFF00000000L | (firstIndex & 0xFFFFFFFFL);
|
|
firstIndex = startPos;
|
|
}
|
|
|
|
protected void moveToLastIndex(int startPos) {
|
|
if(size == 1 || lastIndex == startPos) return;
|
|
if(firstIndex == startPos) {
|
|
firstIndex = (int)links[startPos];
|
|
links[lastIndex] |= 0xFFFFFFFF00000000L;
|
|
}
|
|
else {
|
|
long link = links[startPos];
|
|
int prev = (int)(link >>> 32);
|
|
int next = (int)link;
|
|
links[prev] ^= ((links[prev] ^ (link & 0xFFFFFFFFL)) & 0xFFFFFFFFL);
|
|
links[next] ^= ((links[next] ^ (link & 0xFFFFFFFF00000000L)) & 0xFFFFFFFF00000000L);
|
|
}
|
|
links[lastIndex] ^= ((links[lastIndex] ^ (startPos & 0xFFFFFFFFL)) & 0xFFFFFFFFL);
|
|
links[startPos] = ((lastIndex & 0xFFFFFFFFL) << 32) | 0xFFFFFFFFL;
|
|
lastIndex = startPos;
|
|
}
|
|
|
|
@Override
|
|
protected void onNodeAdded(int pos) {
|
|
if(size == 0) {
|
|
firstIndex = lastIndex = pos;
|
|
links[pos] = -1L;
|
|
}
|
|
else {
|
|
links[lastIndex] ^= ((links[lastIndex] ^ (pos & 0xFFFFFFFFL)) & 0xFFFFFFFFL);
|
|
links[pos] = ((lastIndex & 0xFFFFFFFFL) << 32) | 0xFFFFFFFFL;
|
|
lastIndex = pos;
|
|
}
|
|
}
|
|
|
|
@Override
|
|
protected void onNodeRemoved(int pos) {
|
|
if(size == 0) firstIndex = lastIndex = -1;
|
|
else if(firstIndex == pos) {
|
|
firstIndex = (int)links[pos];
|
|
if(0 <= firstIndex) links[firstIndex] |= 0xFFFFFFFF00000000L;
|
|
}
|
|
else if(lastIndex == pos) {
|
|
lastIndex = (int)(links[pos] >>> 32);
|
|
if(0 <= lastIndex) links[lastIndex] |= 0xFFFFFFFFL;
|
|
}
|
|
else {
|
|
long link = links[pos];
|
|
int prev = (int)(link >>> 32);
|
|
int next = (int)link;
|
|
links[prev] ^= ((links[prev] ^ (link & 0xFFFFFFFFL)) & 0xFFFFFFFFL);
|
|
links[next] ^= ((links[next] ^ (link & 0xFFFFFFFF00000000L)) & 0xFFFFFFFF00000000L);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
protected void onNodeMoved(int from, int to) {
|
|
if(size == 1) {
|
|
firstIndex = lastIndex = to;
|
|
links[to] = -1L;
|
|
}
|
|
else if(firstIndex == from) {
|
|
firstIndex = to;
|
|
links[(int)links[from]] ^= ((links[(int)links[from]] ^ ((to & 0xFFFFFFFFL) << 32)) & 0xFFFFFFFF00000000L);
|
|
links[to] = links[from];
|
|
}
|
|
else if(lastIndex == from) {
|
|
lastIndex = to;
|
|
links[(int)(links[from] >>> 32)] ^= ((links[(int)(links[from] >>> 32)] ^ (to & 0xFFFFFFFFL)) & 0xFFFFFFFFL);
|
|
links[to] = links[from];
|
|
}
|
|
else {
|
|
long link = links[from];
|
|
int prev = (int)(link >>> 32);
|
|
int next = (int)link;
|
|
links[prev] ^= ((links[prev] ^ (to & 0xFFFFFFFFL)) & 0xFFFFFFFFL);
|
|
links[next] ^= ((links[next] ^ ((to & 0xFFFFFFFFL) << 32)) & 0xFFFFFFFF00000000L);
|
|
links[to] = link;
|
|
}
|
|
}
|
|
|
|
@Override
|
|
protected void rehash(int newSize) {
|
|
int newMask = newSize - 1;
|
|
KEY_TYPE[] newKeys = NEW_KEY_ARRAY(newSize + 1);
|
|
VALUE_TYPE[] newValues = NEW_VALUE_ARRAY(newSize + 1);
|
|
long[] newLinks = new long[newSize + 1];
|
|
int i = firstIndex, prev = -1, newPrev = -1, pos;
|
|
firstIndex = -1;
|
|
for(int j = size; j-- != 0;) {
|
|
if(strategy.equals(keys[i], EMPTY_KEY_VALUE)) pos = newSize;
|
|
else {
|
|
pos = HashUtil.mix(strategy.hashCode(keys[i])) & newMask;
|
|
while(!strategy.equals(newKeys[pos], EMPTY_KEY_VALUE)) pos = ++pos & newMask;
|
|
}
|
|
newKeys[pos] = keys[i];
|
|
newValues[pos] = values[i];
|
|
if(prev != -1) {
|
|
newLinks[newPrev] ^= ((newLinks[newPrev] ^ (pos & 0xFFFFFFFFL)) & 0xFFFFFFFFL);
|
|
newLinks[pos] ^= ((newLinks[pos] ^ ((newPrev & 0xFFFFFFFFL) << 32)) & 0xFFFFFFFF00000000L);
|
|
newPrev = pos;
|
|
}
|
|
else {
|
|
newPrev = firstIndex = pos;
|
|
newLinks[pos] = -1L;
|
|
}
|
|
i = (int)links[prev = i];
|
|
}
|
|
links = newLinks;
|
|
lastIndex = newPrev;
|
|
if(newPrev != -1) newLinks[newPrev] |= 0xFFFFFFFFL;
|
|
nullIndex = newSize;
|
|
mask = newMask;
|
|
maxFill = Math.min((int)Math.ceil(nullIndex * loadFactor), nullIndex - 1);
|
|
keys = newKeys;
|
|
values = newValues;
|
|
}
|
|
|
|
private class MapEntrySet extends AbstractObjectSet<MAP.Entry KEY_VALUE_GENERIC_TYPE> implements ORDERED_MAP.FastOrderedSet KEY_VALUE_GENERIC_TYPE {
|
|
@Override
|
|
public boolean addAndMoveToFirst(MAP.Entry KEY_VALUE_GENERIC_TYPE o) { throw new UnsupportedOperationException(); }
|
|
@Override
|
|
public boolean addAndMoveToLast(MAP.Entry KEY_VALUE_GENERIC_TYPE o) { throw new UnsupportedOperationException(); }
|
|
|
|
@Override
|
|
public boolean moveToFirst(MAP.Entry KEY_VALUE_GENERIC_TYPE o) {
|
|
return LINKED_CUSTOM_HASH_MAP.this.moveToFirst(o.ENTRY_KEY());
|
|
}
|
|
|
|
@Override
|
|
public boolean moveToLast(MAP.Entry KEY_VALUE_GENERIC_TYPE o) {
|
|
return LINKED_CUSTOM_HASH_MAP.this.moveToLast(o.ENTRY_KEY());
|
|
}
|
|
|
|
@Override
|
|
public MAP.Entry KEY_VALUE_GENERIC_TYPE first() {
|
|
return new BasicEntryKV_BRACES(FIRST_ENTRY_KEY(), FIRST_ENTRY_VALUE());
|
|
}
|
|
|
|
@Override
|
|
public MAP.Entry KEY_VALUE_GENERIC_TYPE last() {
|
|
return new BasicEntryKV_BRACES(LAST_ENTRY_KEY(), LAST_ENTRY_VALUE());
|
|
}
|
|
|
|
@Override
|
|
public MAP.Entry KEY_VALUE_GENERIC_TYPE pollFirst() {
|
|
BasicEntry KEY_VALUE_GENERIC_TYPE entry = new BasicEntryKV_BRACES(FIRST_ENTRY_KEY(), FIRST_ENTRY_VALUE());
|
|
POLL_FIRST_ENTRY_KEY();
|
|
return entry;
|
|
}
|
|
|
|
@Override
|
|
public MAP.Entry KEY_VALUE_GENERIC_TYPE pollLast() {
|
|
BasicEntry KEY_VALUE_GENERIC_TYPE entry = new BasicEntryKV_BRACES(LAST_ENTRY_KEY(), LAST_ENTRY_VALUE());
|
|
POLL_LAST_ENTRY_KEY();
|
|
return entry;
|
|
}
|
|
|
|
@Override
|
|
public ObjectBidirectionalIterator<MAP.Entry KEY_VALUE_GENERIC_TYPE> iterator() {
|
|
return new EntryIterator();
|
|
}
|
|
|
|
@Override
|
|
public ObjectBidirectionalIterator<MAP.Entry KEY_VALUE_GENERIC_TYPE> iterator(MAP.Entry KEY_VALUE_GENERIC_TYPE fromElement) {
|
|
return new EntryIterator(fromElement.ENTRY_KEY());
|
|
}
|
|
|
|
@Override
|
|
public ObjectBidirectionalIterator<MAP.Entry KEY_VALUE_GENERIC_TYPE> fastIterator() {
|
|
return new FastEntryIterator();
|
|
}
|
|
|
|
@Override
|
|
public ObjectBidirectionalIterator<MAP.Entry KEY_VALUE_GENERIC_TYPE> fastIterator(KEY_TYPE fromElement) {
|
|
return new FastEntryIterator(fromElement);
|
|
}
|
|
|
|
@Override
|
|
public MapEntrySet copy() { throw new UnsupportedOperationException(); }
|
|
|
|
@Override
|
|
public void forEach(Consumer<? super MAP.Entry KEY_VALUE_GENERIC_TYPE> action) {
|
|
int index = firstIndex;
|
|
while(index != -1) {
|
|
action.accept(new ValueMapEntry(index));
|
|
index = (int)links[index];
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void fastForEach(Consumer<? super MAP.Entry KEY_VALUE_GENERIC_TYPE> action) {
|
|
MapEntry entry = new MapEntry();
|
|
int index = firstIndex;
|
|
while(index != -1) {
|
|
entry.set(index);
|
|
action.accept(entry);
|
|
index = (int)links[index];
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void forEachIndexed(IntObjectConsumer<MAP.Entry KEY_VALUE_GENERIC_TYPE> action) {
|
|
Objects.requireNonNull(action);
|
|
if(size() <= 0) return;
|
|
int count = 0;
|
|
int index = firstIndex;
|
|
while(index != -1) {
|
|
action.accept(count++, new ValueMapEntry(index));
|
|
index = (int)links[index];
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public <E> void forEach(E input, ObjectObjectConsumer<E, MAP.Entry KEY_VALUE_GENERIC_TYPE> action) {
|
|
Objects.requireNonNull(action);
|
|
if(size() <= 0) return;
|
|
int index = firstIndex;
|
|
while(index != -1) {
|
|
action.accept(input, new ValueMapEntry(index));
|
|
index = (int)links[index];
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public boolean matchesAny(Predicate<MAP.Entry KEY_VALUE_GENERIC_TYPE> filter) {
|
|
Objects.requireNonNull(filter);
|
|
if(size() <= 0) return false;
|
|
MapEntry entry = new MapEntry();
|
|
int index = firstIndex;
|
|
while(index != -1) {
|
|
entry.set(index);
|
|
if(filter.test(entry)) return true;
|
|
index = (int)links[index];
|
|
}
|
|
return false;
|
|
}
|
|
|
|
@Override
|
|
public boolean matchesNone(Predicate<MAP.Entry KEY_VALUE_GENERIC_TYPE> filter) {
|
|
Objects.requireNonNull(filter);
|
|
if(size() <= 0) return true;
|
|
MapEntry entry = new MapEntry();
|
|
int index = firstIndex;
|
|
while(index != -1) {
|
|
entry.set(index);
|
|
if(filter.test(entry)) return false;
|
|
index = (int)links[index];
|
|
}
|
|
return true;
|
|
}
|
|
|
|
@Override
|
|
public boolean matchesAll(Predicate<MAP.Entry KEY_VALUE_GENERIC_TYPE> filter) {
|
|
Objects.requireNonNull(filter);
|
|
if(size() <= 0) return true;
|
|
MapEntry entry = new MapEntry();
|
|
int index = firstIndex;
|
|
while(index != -1) {
|
|
entry.set(index);
|
|
if(!filter.test(entry)) return false;
|
|
index = (int)links[index];
|
|
}
|
|
return true;
|
|
}
|
|
|
|
@Override
|
|
public <E> E reduce(E identity, BiFunction<E, MAP.Entry KEY_VALUE_GENERIC_TYPE, E> operator) {
|
|
Objects.requireNonNull(operator);
|
|
E state = identity;
|
|
int index = firstIndex;
|
|
while(index != -1) {
|
|
state = operator.apply(state, new ValueMapEntry(index));
|
|
index = (int)links[index];
|
|
}
|
|
return state;
|
|
}
|
|
|
|
@Override
|
|
public MAP.Entry KEY_VALUE_GENERIC_TYPE reduce(ObjectObjectUnaryOperator<MAP.Entry KEY_VALUE_GENERIC_TYPE, MAP.Entry KEY_VALUE_GENERIC_TYPE> operator) {
|
|
Objects.requireNonNull(operator);
|
|
MAP.Entry KEY_VALUE_GENERIC_TYPE state = null;
|
|
boolean empty = true;
|
|
int index = firstIndex;
|
|
while(index != -1) {
|
|
if(empty) {
|
|
empty = false;
|
|
state = new ValueMapEntry(index);
|
|
index = (int)links[index];
|
|
continue;
|
|
}
|
|
state = operator.apply(state, new ValueMapEntry(index));
|
|
index = (int)links[index];
|
|
}
|
|
return state;
|
|
}
|
|
|
|
@Override
|
|
public MAP.Entry KEY_VALUE_GENERIC_TYPE findFirst(Predicate<MAP.Entry KEY_VALUE_GENERIC_TYPE> filter) {
|
|
Objects.requireNonNull(filter);
|
|
if(size() <= 0) return null;
|
|
MapEntry entry = new MapEntry();
|
|
int index = firstIndex;
|
|
while(index != -1) {
|
|
entry.set(index);
|
|
if(filter.test(entry)) return entry;
|
|
index = (int)links[index];
|
|
}
|
|
return null;
|
|
}
|
|
|
|
@Override
|
|
public int count(Predicate<MAP.Entry KEY_VALUE_GENERIC_TYPE> filter) {
|
|
Objects.requireNonNull(filter);
|
|
if(size() <= 0) return 0;
|
|
MapEntry entry = new MapEntry();
|
|
int result = 0;
|
|
int index = firstIndex;
|
|
while(index != -1) {
|
|
entry.set(index);
|
|
if(filter.test(entry)) result++;
|
|
index = (int)links[index];
|
|
}
|
|
return result;
|
|
}
|
|
|
|
@Override
|
|
@Deprecated
|
|
public boolean contains(Object o) {
|
|
if(o instanceof Map.Entry) {
|
|
if(o instanceof MAP.Entry) {
|
|
MAP.Entry KEY_VALUE_GENERIC_TYPE entry = (MAP.Entry KEY_VALUE_GENERIC_TYPE)o;
|
|
int index = LINKED_CUSTOM_HASH_MAP.this.findIndex(entry.ENTRY_KEY());
|
|
if(index >= 0) return VALUE_EQUALS(entry.ENTRY_VALUE(), LINKED_CUSTOM_HASH_MAP.this.values[index]);
|
|
}
|
|
else {
|
|
Map.Entry<?, ?> entry = (Map.Entry<?, ?>)o;
|
|
#if !TYPE_OBJECT
|
|
if(!(entry.getKey() instanceof CLASS_TYPE)) return false;
|
|
#endif
|
|
int index = LINKED_CUSTOM_HASH_MAP.this.findIndex((CLASS_TYPE)entry.getKey());
|
|
if(index >= 0) return Objects.equals(entry.getValue(), VALUE_TO_OBJ(LINKED_CUSTOM_HASH_MAP.this.values[index]));
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
@Override
|
|
@Deprecated
|
|
public boolean remove(Object o) {
|
|
if(o instanceof Map.Entry) {
|
|
if(o instanceof MAP.Entry) {
|
|
MAP.Entry KEY_VALUE_GENERIC_TYPE entry = (MAP.Entry KEY_VALUE_GENERIC_TYPE)o;
|
|
return LINKED_CUSTOM_HASH_MAP.this.remove(entry.ENTRY_KEY(), entry.ENTRY_VALUE());
|
|
}
|
|
Map.Entry<?, ?> entry = (Map.Entry<?, ?>)o;
|
|
return LINKED_CUSTOM_HASH_MAP.this.remove(entry.getKey(), entry.getValue());
|
|
}
|
|
return false;
|
|
}
|
|
|
|
@Override
|
|
public int size() {
|
|
return LINKED_CUSTOM_HASH_MAP.this.size();
|
|
}
|
|
|
|
@Override
|
|
public void clear() {
|
|
LINKED_CUSTOM_HASH_MAP.this.clear();
|
|
}
|
|
}
|
|
|
|
private final class KeySet extends ABSTRACT_SET KEY_GENERIC_TYPE implements ORDERED_SET KEY_GENERIC_TYPE {
|
|
#if TYPE_OBJECT
|
|
@Override
|
|
@Deprecated
|
|
public boolean contains(Object e) {
|
|
return containsKey(e);
|
|
}
|
|
|
|
@Override
|
|
public boolean remove(Object o) {
|
|
int oldSize = size;
|
|
LINKED_CUSTOM_HASH_MAP.this.remove(o);
|
|
return size != oldSize;
|
|
}
|
|
|
|
#else
|
|
@Override
|
|
public boolean contains(KEY_TYPE e) {
|
|
return containsKey(e);
|
|
}
|
|
|
|
@Override
|
|
public boolean remove(KEY_TYPE o) {
|
|
int oldSize = size;
|
|
LINKED_CUSTOM_HASH_MAP.this.remove(o);
|
|
return size != oldSize;
|
|
}
|
|
|
|
#endif
|
|
@Override
|
|
public boolean add(KEY_TYPE o) {
|
|
throw new UnsupportedOperationException();
|
|
}
|
|
|
|
@Override
|
|
public boolean addAndMoveToFirst(KEY_TYPE o) { throw new UnsupportedOperationException(); }
|
|
|
|
@Override
|
|
public boolean addAndMoveToLast(KEY_TYPE o) { throw new UnsupportedOperationException(); }
|
|
|
|
@Override
|
|
public boolean moveToFirst(KEY_TYPE o) {
|
|
return LINKED_CUSTOM_HASH_MAP.this.moveToFirst(o);
|
|
}
|
|
|
|
@Override
|
|
public boolean moveToLast(KEY_TYPE o) {
|
|
return LINKED_CUSTOM_HASH_MAP.this.moveToLast(o);
|
|
}
|
|
|
|
@Override
|
|
public LIST_ITERATOR KEY_GENERIC_TYPE iterator() {
|
|
return new KeyIterator();
|
|
}
|
|
|
|
@Override
|
|
public BI_ITERATOR KEY_GENERIC_TYPE iterator(KEY_TYPE fromElement) {
|
|
return new KeyIterator(fromElement);
|
|
}
|
|
|
|
@Override
|
|
public KeySet copy() { throw new UnsupportedOperationException(); }
|
|
|
|
@Override
|
|
public int size() {
|
|
return LINKED_CUSTOM_HASH_MAP.this.size();
|
|
}
|
|
|
|
@Override
|
|
public void clear() {
|
|
LINKED_CUSTOM_HASH_MAP.this.clear();
|
|
}
|
|
|
|
@Override
|
|
public KEY_TYPE FIRST_KEY() {
|
|
return FIRST_ENTRY_KEY();
|
|
}
|
|
|
|
@Override
|
|
public KEY_TYPE POLL_FIRST_KEY() {
|
|
return POLL_FIRST_ENTRY_KEY();
|
|
}
|
|
|
|
@Override
|
|
public KEY_TYPE LAST_KEY() {
|
|
return LAST_ENTRY_KEY();
|
|
}
|
|
|
|
@Override
|
|
public KEY_TYPE POLL_LAST_KEY() {
|
|
return POLL_LAST_ENTRY_KEY();
|
|
}
|
|
|
|
@Override
|
|
public void forEach(CONSUMER KEY_SUPER_GENERIC_TYPE action) {
|
|
int index = firstIndex;
|
|
while(index != -1){
|
|
action.accept(keys[index]);
|
|
index = (int)links[index];
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void forEachIndexed(BI_FROM_INT_CONSUMER KEY_GENERIC_TYPE action) {
|
|
Objects.requireNonNull(action);
|
|
if(size() <= 0) return;
|
|
int count = 0;
|
|
int index = firstIndex;
|
|
while(index != -1){
|
|
action.accept(count++, keys[index]);
|
|
index = (int)links[index];
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public <E> void forEach(E input, BI_FROM_OBJECT_CONSUMER KSK_GENERIC_TYPE<E> action) {
|
|
Objects.requireNonNull(action);
|
|
if(size() <= 0) return;
|
|
int index = firstIndex;
|
|
while(index != -1){
|
|
action.accept(input, keys[index]);
|
|
index = (int)links[index];
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public boolean matchesAny(PREDICATE KEY_GENERIC_TYPE filter) {
|
|
Objects.requireNonNull(filter);
|
|
if(size() <= 0) return false;
|
|
int index = firstIndex;
|
|
while(index != -1){
|
|
if(filter.test(keys[index])) return true;
|
|
index = (int)links[index];
|
|
}
|
|
return false;
|
|
}
|
|
|
|
@Override
|
|
public boolean matchesNone(PREDICATE KEY_GENERIC_TYPE filter) {
|
|
Objects.requireNonNull(filter);
|
|
if(size() <= 0) return true;
|
|
int index = firstIndex;
|
|
while(index != -1){
|
|
if(filter.test(keys[index])) return false;
|
|
index = (int)links[index];
|
|
}
|
|
return true;
|
|
}
|
|
|
|
@Override
|
|
public boolean matchesAll(PREDICATE KEY_GENERIC_TYPE filter) {
|
|
Objects.requireNonNull(filter);
|
|
if(size() <= 0) return true;
|
|
int index = firstIndex;
|
|
while(index != -1){
|
|
if(!filter.test(keys[index])) return false;
|
|
index = (int)links[index];
|
|
}
|
|
return true;
|
|
}
|
|
|
|
#if !TYPE_OBJECT
|
|
@Override
|
|
public KEY_TYPE reduce(KEY_TYPE identity, SINGLE_UNARY_OPERATOR KEY_KEY_GENERIC_TYPE operator) {
|
|
Objects.requireNonNull(operator);
|
|
KEY_TYPE state = identity;
|
|
int index = firstIndex;
|
|
while(index != -1) {
|
|
state = operator.APPLY_KEY_VALUE(state, keys[index]);
|
|
index = (int)links[index];
|
|
}
|
|
return state;
|
|
}
|
|
|
|
#else
|
|
@Override
|
|
public <KEY_SPECIAL_TYPE> KEY_SPECIAL_TYPE reduce(KEY_SPECIAL_TYPE identity, BiFunction<KEY_SPECIAL_TYPE, KEY_TYPE, KEY_SPECIAL_TYPE> operator) {
|
|
Objects.requireNonNull(operator);
|
|
KEY_SPECIAL_TYPE state = identity;
|
|
int index = firstIndex;
|
|
while(index != -1) {
|
|
state = operator.apply(state, keys[index]);
|
|
index = (int)links[index];
|
|
}
|
|
return state;
|
|
}
|
|
|
|
#endif
|
|
@Override
|
|
public KEY_TYPE reduce(SINGLE_UNARY_OPERATOR KEY_KEY_GENERIC_TYPE operator) {
|
|
Objects.requireNonNull(operator);
|
|
KEY_TYPE state = EMPTY_KEY_VALUE;
|
|
boolean empty = true;
|
|
int index = firstIndex;
|
|
while(index != -1) {
|
|
if(empty) {
|
|
empty = false;
|
|
state = keys[index];
|
|
index = (int)links[index];
|
|
continue;
|
|
}
|
|
state = operator.APPLY_KEY_VALUE(state, keys[index]);
|
|
index = (int)links[index];
|
|
}
|
|
return state;
|
|
}
|
|
|
|
@Override
|
|
public KEY_TYPE findFirst(PREDICATE KEY_GENERIC_TYPE filter) {
|
|
Objects.requireNonNull(filter);
|
|
if(size() <= 0) return EMPTY_KEY_VALUE;
|
|
int index = firstIndex;
|
|
while(index != -1){
|
|
if(filter.test(keys[index])) return keys[index];
|
|
index = (int)links[index];
|
|
}
|
|
return EMPTY_KEY_VALUE;
|
|
}
|
|
|
|
@Override
|
|
public int count(PREDICATE KEY_GENERIC_TYPE filter) {
|
|
Objects.requireNonNull(filter);
|
|
if(size() <= 0) return 0;
|
|
int result = 0;
|
|
int index = firstIndex;
|
|
while(index != -1){
|
|
if(filter.test(keys[index])) result++;
|
|
index = (int)links[index];
|
|
}
|
|
return result;
|
|
}
|
|
}
|
|
|
|
private class Values extends VALUE_ABSTRACT_COLLECTION VALUE_GENERIC_TYPE {
|
|
#if VALUE_OBJECT
|
|
@Override
|
|
@Deprecated
|
|
public boolean contains(Object e) {
|
|
return containsValue(e);
|
|
}
|
|
|
|
#else
|
|
@Override
|
|
public boolean contains(VALUE_TYPE e) {
|
|
return containsValue(e);
|
|
}
|
|
|
|
#endif
|
|
@Override
|
|
public boolean add(VALUE_TYPE o) {
|
|
throw new UnsupportedOperationException();
|
|
}
|
|
|
|
@Override
|
|
public VALUE_ITERATOR VALUE_GENERIC_TYPE iterator() {
|
|
return new ValueIterator();
|
|
}
|
|
|
|
@Override
|
|
public int size() {
|
|
return LINKED_CUSTOM_HASH_MAP.this.size();
|
|
}
|
|
|
|
@Override
|
|
public void clear() {
|
|
LINKED_CUSTOM_HASH_MAP.this.clear();
|
|
}
|
|
|
|
@Override
|
|
public void forEach(VALUE_CONSUMER VALUE_SUPER_GENERIC_TYPE action) {
|
|
int index = firstIndex;
|
|
while(index != -1){
|
|
action.accept(values[index]);
|
|
index = (int)links[index];
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void forEachIndexed(VALUE_BI_FROM_INT_CONSUMER VALUE_GENERIC_TYPE action) {
|
|
Objects.requireNonNull(action);
|
|
if(size() <= 0) return;
|
|
int count = 0;
|
|
int index = firstIndex;
|
|
while(index != -1){
|
|
action.accept(count++, values[index]);
|
|
index = (int)links[index];
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public <E> void forEach(E input, VALUE_BI_FROM_OBJECT_CONSUMER VSV_GENERIC_TYPE<E> action) {
|
|
Objects.requireNonNull(action);
|
|
if(size() <= 0) return;
|
|
int index = firstIndex;
|
|
while(index != -1){
|
|
action.accept(input, values[index]);
|
|
index = (int)links[index];
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public boolean matchesAny(VALUE_PREDICATE VALUE_GENERIC_TYPE filter) {
|
|
Objects.requireNonNull(filter);
|
|
if(size() <= 0) return false;
|
|
int index = firstIndex;
|
|
while(index != -1){
|
|
if(filter.test(values[index])) return true;
|
|
index = (int)links[index];
|
|
}
|
|
return false;
|
|
}
|
|
|
|
@Override
|
|
public boolean matchesNone(VALUE_PREDICATE VALUE_GENERIC_TYPE filter) {
|
|
Objects.requireNonNull(filter);
|
|
if(size() <= 0) return true;
|
|
int index = firstIndex;
|
|
while(index != -1){
|
|
if(filter.test(values[index])) return false;
|
|
index = (int)links[index];
|
|
}
|
|
return true;
|
|
}
|
|
|
|
@Override
|
|
public boolean matchesAll(VALUE_PREDICATE VALUE_GENERIC_TYPE filter) {
|
|
Objects.requireNonNull(filter);
|
|
if(size() <= 0) return true;
|
|
int index = firstIndex;
|
|
while(index != -1){
|
|
if(!filter.test(values[index])) return false;
|
|
index = (int)links[index];
|
|
}
|
|
return true;
|
|
}
|
|
|
|
#if !VALUE_OBJECT
|
|
@Override
|
|
public VALUE_TYPE reduce(VALUE_TYPE identity, VALUE_SINGLE_UNARY_OPERATOR VALUE_VALUE_GENERIC_TYPE operator) {
|
|
Objects.requireNonNull(operator);
|
|
VALUE_TYPE state = identity;
|
|
int index = firstIndex;
|
|
while(index != -1) {
|
|
state = operator.APPLY_VALUE(state, values[index]);
|
|
index = (int)links[index];
|
|
}
|
|
return state;
|
|
}
|
|
|
|
#else
|
|
@Override
|
|
public <VALUE_SPECIAL_TYPE> VALUE_SPECIAL_TYPE reduce(VALUE_SPECIAL_TYPE identity, BiFunction<VALUE_SPECIAL_TYPE, VALUE_TYPE, VALUE_SPECIAL_TYPE> operator) {
|
|
Objects.requireNonNull(operator);
|
|
VALUE_SPECIAL_TYPE state = identity;
|
|
int index = firstIndex;
|
|
while(index != -1) {
|
|
state = operator.apply(state, values[index]);
|
|
index = (int)links[index];
|
|
}
|
|
return state;
|
|
}
|
|
|
|
#endif
|
|
@Override
|
|
public VALUE_TYPE reduce(VALUE_SINGLE_UNARY_OPERATOR VALUE_VALUE_GENERIC_TYPE operator) {
|
|
Objects.requireNonNull(operator);
|
|
VALUE_TYPE state = EMPTY_VALUE;
|
|
boolean empty = true;
|
|
int index = firstIndex;
|
|
while(index != -1) {
|
|
if(empty) {
|
|
empty = false;
|
|
state = values[index];
|
|
index = (int)links[index];
|
|
continue;
|
|
}
|
|
state = operator.APPLY_VALUE(state, values[index]);
|
|
index = (int)links[index];
|
|
}
|
|
return state;
|
|
}
|
|
|
|
@Override
|
|
public VALUE_TYPE findFirst(VALUE_PREDICATE VALUE_GENERIC_TYPE filter) {
|
|
Objects.requireNonNull(filter);
|
|
if(size() <= 0) return EMPTY_VALUE;
|
|
int index = firstIndex;
|
|
while(index != -1){
|
|
if(filter.test(values[index])) return values[index];
|
|
index = (int)links[index];
|
|
}
|
|
return EMPTY_VALUE;
|
|
}
|
|
|
|
@Override
|
|
public int count(VALUE_PREDICATE VALUE_GENERIC_TYPE filter) {
|
|
Objects.requireNonNull(filter);
|
|
if(size() <= 0) return 0;
|
|
int result = 0;
|
|
int index = firstIndex;
|
|
while(index != -1){
|
|
if(filter.test(values[index])) result++;
|
|
index = (int)links[index];
|
|
}
|
|
return result;
|
|
}
|
|
}
|
|
|
|
private class FastEntryIterator extends MapIterator implements ObjectListIterator<MAP.Entry KEY_VALUE_GENERIC_TYPE> {
|
|
MapEntry entry = new MapEntry();
|
|
|
|
public FastEntryIterator() {}
|
|
public FastEntryIterator(KEY_TYPE from) {
|
|
super(from);
|
|
}
|
|
|
|
@Override
|
|
public MAP.Entry KEY_VALUE_GENERIC_TYPE next() {
|
|
entry.index = nextEntry();
|
|
return entry;
|
|
}
|
|
|
|
@Override
|
|
public MAP.Entry KEY_VALUE_GENERIC_TYPE previous() {
|
|
entry.index = previousEntry();
|
|
return entry;
|
|
}
|
|
|
|
@Override
|
|
public void set(MAP.Entry KEY_VALUE_GENERIC_TYPE entry) { throw new UnsupportedOperationException(); }
|
|
|
|
@Override
|
|
public void add(MAP.Entry KEY_VALUE_GENERIC_TYPE entry) { throw new UnsupportedOperationException(); }
|
|
}
|
|
|
|
private class EntryIterator extends MapIterator implements ObjectListIterator<MAP.Entry KEY_VALUE_GENERIC_TYPE> {
|
|
MapEntry entry;
|
|
|
|
public EntryIterator() {}
|
|
public EntryIterator(KEY_TYPE from) {
|
|
super(from);
|
|
}
|
|
|
|
@Override
|
|
public MAP.Entry KEY_VALUE_GENERIC_TYPE next() {
|
|
return entry = new ValueMapEntry(nextEntry());
|
|
}
|
|
|
|
@Override
|
|
public MAP.Entry KEY_VALUE_GENERIC_TYPE previous() {
|
|
return entry = new ValueMapEntry(previousEntry());
|
|
}
|
|
|
|
@Override
|
|
public void remove() {
|
|
super.remove();
|
|
entry.index = -1;
|
|
}
|
|
|
|
@Override
|
|
public void set(MAP.Entry KEY_VALUE_GENERIC_TYPE entry) { throw new UnsupportedOperationException(); }
|
|
|
|
@Override
|
|
public void add(MAP.Entry KEY_VALUE_GENERIC_TYPE entry) { throw new UnsupportedOperationException(); }
|
|
}
|
|
|
|
private class KeyIterator extends MapIterator implements LIST_ITERATOR KEY_GENERIC_TYPE {
|
|
|
|
public KeyIterator() {}
|
|
public KeyIterator(KEY_TYPE from) {
|
|
super(from);
|
|
}
|
|
|
|
@Override
|
|
public KEY_TYPE PREVIOUS() {
|
|
return keys[previousEntry()];
|
|
}
|
|
|
|
@Override
|
|
public KEY_TYPE NEXT() {
|
|
return keys[nextEntry()];
|
|
}
|
|
|
|
@Override
|
|
public void set(KEY_TYPE e) { throw new UnsupportedOperationException(); }
|
|
@Override
|
|
public void add(KEY_TYPE e) { throw new UnsupportedOperationException(); }
|
|
}
|
|
|
|
private class ValueIterator extends MapIterator implements VALUE_LIST_ITERATOR VALUE_GENERIC_TYPE {
|
|
public ValueIterator() {}
|
|
|
|
@Override
|
|
public VALUE_TYPE VALUE_PREVIOUS() {
|
|
return values[previousEntry()];
|
|
}
|
|
|
|
@Override
|
|
public VALUE_TYPE VALUE_NEXT() {
|
|
return values[nextEntry()];
|
|
}
|
|
|
|
@Override
|
|
public void set(VALUE_TYPE e) { throw new UnsupportedOperationException(); }
|
|
|
|
@Override
|
|
public void add(VALUE_TYPE e) { throw new UnsupportedOperationException(); }
|
|
|
|
}
|
|
|
|
private class MapIterator {
|
|
int previous = -1;
|
|
int next = -1;
|
|
int current = -1;
|
|
int index = 0;
|
|
|
|
MapIterator() {
|
|
next = firstIndex;
|
|
}
|
|
|
|
MapIterator(KEY_TYPE from) {
|
|
if(strategy.equals(from, EMPTY_KEY_VALUE)) {
|
|
if(containsNull) {
|
|
next = (int) links[nullIndex];
|
|
previous = nullIndex;
|
|
}
|
|
else throw new NoSuchElementException("The null element is not in the set");
|
|
}
|
|
else if(keys[lastIndex] == from) {
|
|
previous = lastIndex;
|
|
index = size;
|
|
}
|
|
else {
|
|
int pos = HashUtil.mix(strategy.hashCode(from)) & mask;
|
|
while(!strategy.equals(keys[pos], EMPTY_KEY_VALUE)) {
|
|
if(strategy.equals(keys[pos], from)) {
|
|
next = (int)links[pos];
|
|
previous = pos;
|
|
break;
|
|
}
|
|
pos = ++pos & mask;
|
|
}
|
|
if(previous == -1 && next == -1)
|
|
throw new NoSuchElementException("The element was not found");
|
|
}
|
|
}
|
|
|
|
public boolean hasNext() {
|
|
return next != -1;
|
|
}
|
|
|
|
public boolean hasPrevious() {
|
|
return previous != -1;
|
|
}
|
|
|
|
public int nextIndex() {
|
|
ensureIndexKnown();
|
|
return index;
|
|
}
|
|
|
|
public int previousIndex() {
|
|
ensureIndexKnown();
|
|
return index - 1;
|
|
}
|
|
|
|
public void remove() {
|
|
if(current == -1) throw new IllegalStateException();
|
|
ensureIndexKnown();
|
|
if(current == previous) {
|
|
index--;
|
|
previous = (int)(links[current] >>> 32);
|
|
}
|
|
else next = (int)links[current];
|
|
size--;
|
|
if(previous == -1) firstIndex = next;
|
|
else links[previous] ^= ((links[previous] ^ (next & 0xFFFFFFFFL)) & 0xFFFFFFFFL);
|
|
|
|
if (next == -1) lastIndex = previous;
|
|
else links[next] ^= ((links[next] ^ ((previous & 0xFFFFFFFFL) << 32)) & 0xFFFFFFFF00000000L);
|
|
if(current == nullIndex) {
|
|
current = -1;
|
|
containsNull = false;
|
|
keys[nullIndex] = EMPTY_KEY_VALUE;
|
|
values[nullIndex] = EMPTY_VALUE;
|
|
}
|
|
else {
|
|
int slot, last, startPos = current;
|
|
current = -1;
|
|
KEY_TYPE current;
|
|
while(true) {
|
|
startPos = ((last = startPos) + 1) & mask;
|
|
while(true){
|
|
if(strategy.equals((current = keys[startPos]), EMPTY_KEY_VALUE)) {
|
|
keys[last] = EMPTY_KEY_VALUE;
|
|
values[last] = EMPTY_VALUE;
|
|
return;
|
|
}
|
|
slot = HashUtil.mix(strategy.hashCode(current)) & mask;
|
|
if(last <= startPos ? (last >= slot || slot > startPos) : (last >= slot && slot > startPos)) break;
|
|
startPos = ++startPos & mask;
|
|
}
|
|
keys[last] = current;
|
|
values[last] = values[startPos];
|
|
if(next == startPos) next = last;
|
|
if(previous == startPos) previous = last;
|
|
onNodeMoved(startPos, last);
|
|
}
|
|
}
|
|
}
|
|
|
|
public int previousEntry() {
|
|
if(!hasPrevious()) throw new NoSuchElementException();
|
|
current = previous;
|
|
previous = (int)(links[current] >> 32);
|
|
next = current;
|
|
if(index >= 0) index--;
|
|
return current;
|
|
}
|
|
|
|
public int nextEntry() {
|
|
if(!hasNext()) throw new NoSuchElementException();
|
|
current = next;
|
|
next = (int)(links[current]);
|
|
previous = current;
|
|
if(index >= 0) index++;
|
|
return current;
|
|
}
|
|
|
|
private void ensureIndexKnown() {
|
|
if(index == -1) {
|
|
if(previous == -1) {
|
|
index = 0;
|
|
}
|
|
else if(next == -1) {
|
|
index = size;
|
|
}
|
|
else {
|
|
index = 1;
|
|
for(int pos = firstIndex;pos != previous;pos = (int)links[pos], index++);
|
|
}
|
|
}
|
|
}
|
|
|
|
}
|
|
} |