io.rsocket.loadbalance.Int2LongHashMap Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of rsocket-core Show documentation
Show all versions of rsocket-core Show documentation
Core functionality for the RSocket library
/*
* Copyright 2014-2020 Real Logic Limited.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package io.rsocket.loadbalance;
import java.io.Serializable;
import java.util.AbstractCollection;
import java.util.AbstractSet;
import java.util.Arrays;
import java.util.Iterator;
import java.util.Map;
import java.util.NoSuchElementException;
import java.util.Objects;
import java.util.function.Function;
import java.util.function.IntToLongFunction;
import reactor.util.annotation.Nullable;
/** A open addressing with linear probing hash map specialised for primitive key and value pairs. */
class Int2LongHashMap implements Map, Serializable {
static final float DEFAULT_LOAD_FACTOR = 0.55f;
static final int MIN_CAPACITY = 8;
private static final long serialVersionUID = -690554872053575793L;
private final float loadFactor;
private final long missingValue;
private int resizeThreshold;
private int size = 0;
private final boolean shouldAvoidAllocation;
private long[] entries;
private KeySet keySet;
private ValueCollection values;
private EntrySet entrySet;
/** @param missingValue for the map that represents null. */
public Int2LongHashMap(final long missingValue) {
this(MIN_CAPACITY, DEFAULT_LOAD_FACTOR, missingValue);
}
/**
* @param initialCapacity for the map to override {@link #MIN_CAPACITY}
* @param loadFactor for the map to override {@link #DEFAULT_LOAD_FACTOR}.
* @param missingValue for the map that represents null.
*/
public Int2LongHashMap(
final int initialCapacity, final float loadFactor, final long missingValue) {
this(initialCapacity, loadFactor, missingValue, true);
}
/**
* @param initialCapacity for the map to override {@link #MIN_CAPACITY}
* @param loadFactor for the map to override {@link #DEFAULT_LOAD_FACTOR}.
* @param missingValue for the map that represents null.
* @param shouldAvoidAllocation should allocation be avoided by caching iterators and map entries.
*/
public Int2LongHashMap(
final int initialCapacity,
final float loadFactor,
final long missingValue,
final boolean shouldAvoidAllocation) {
validateLoadFactor(loadFactor);
this.loadFactor = loadFactor;
this.missingValue = missingValue;
this.shouldAvoidAllocation = shouldAvoidAllocation;
capacity(findNextPositivePowerOfTwo(Math.max(MIN_CAPACITY, initialCapacity)));
}
/**
* The value to be used as a null marker in the map.
*
* @return value to be used as a null marker in the map.
*/
public long missingValue() {
return missingValue;
}
/**
* Get the load factor applied for resize operations.
*
* @return the load factor applied for resize operations.
*/
public float loadFactor() {
return loadFactor;
}
/**
* Get the total capacity for the map to which the load factor will be a fraction of.
*
* @return the total capacity for the map.
*/
public int capacity() {
return entries.length >> 1;
}
/**
* Get the actual threshold which when reached the map will resize. This is a function of the
* current capacity and load factor.
*
* @return the threshold when the map will resize.
*/
public int resizeThreshold() {
return resizeThreshold;
}
/** {@inheritDoc} */
public int size() {
return size;
}
/** {@inheritDoc} */
public boolean isEmpty() {
return size == 0;
}
/**
* Get a value using provided key avoiding boxing.
*
* @param key lookup key.
* @return value associated with the key or {@link #missingValue()} if key is not found in the
* map.
*/
public long get(final int key) {
final int mask = entries.length - 1;
int index = evenHash(key, mask);
long value = missingValue;
while (entries[index + 1] != missingValue) {
if (entries[index] == key) {
value = entries[index + 1];
break;
}
index = next(index, mask);
}
return value;
}
/**
* Put a key value pair in the map.
*
* @param key lookup key
* @param value new value, must not be {@link #missingValue()}
* @return previous value associated with the key, or {@link #missingValue()} if none found
* @throws IllegalArgumentException if value is {@link #missingValue()}
*/
public long put(final int key, final long value) {
if (value == missingValue) {
throw new IllegalArgumentException("cannot accept missingValue");
}
final int mask = entries.length - 1;
int index = evenHash(key, mask);
long oldValue = missingValue;
while (entries[index + 1] != missingValue) {
if (entries[index] == key) {
oldValue = entries[index + 1];
break;
}
index = next(index, mask);
}
if (oldValue == missingValue) {
++size;
entries[index] = key;
}
entries[index + 1] = value;
increaseCapacity();
return oldValue;
}
private void increaseCapacity() {
if (size > resizeThreshold) {
// entries.length = 2 * capacity
final int newCapacity = entries.length;
rehash(newCapacity);
}
}
private void rehash(final int newCapacity) {
final long[] oldEntries = entries;
final int length = entries.length;
capacity(newCapacity);
final long[] newEntries = entries;
final int mask = entries.length - 1;
for (int keyIndex = 0; keyIndex < length; keyIndex += 2) {
final long value = oldEntries[keyIndex + 1];
if (value != missingValue) {
final int key = (int) oldEntries[keyIndex];
int index = evenHash(key, mask);
while (newEntries[index + 1] != missingValue) {
index = next(index, mask);
}
newEntries[index] = key;
newEntries[index + 1] = value;
}
}
}
/**
* Int primitive specialised containsKey.
*
* @param key the key to check.
* @return true if the map contains key as a key, false otherwise.
*/
public boolean containsKey(final int key) {
return get(key) != missingValue;
}
/**
* Does the map contain the value.
*
* @param value to be tested against contained values.
* @return true if contained otherwise value.
*/
public boolean containsValue(final long value) {
boolean found = false;
if (value != missingValue) {
final int length = entries.length;
int remaining = size;
for (int valueIndex = 1; remaining > 0 && valueIndex < length; valueIndex += 2) {
if (missingValue != entries[valueIndex]) {
if (value == entries[valueIndex]) {
found = true;
break;
}
--remaining;
}
}
}
return found;
}
/** {@inheritDoc} */
public void clear() {
if (size > 0) {
Arrays.fill(entries, missingValue);
size = 0;
}
}
/**
* Compact the backing arrays by rehashing with a capacity just larger than current size and
* giving consideration to the load factor.
*/
public void compact() {
final int idealCapacity = (int) Math.round(size() * (1.0d / loadFactor));
rehash(findNextPositivePowerOfTwo(Math.max(MIN_CAPACITY, idealCapacity)));
}
/**
* Primitive specialised version of {@link #computeIfAbsent(Object, Function)}
*
* @param key to search on.
* @param mappingFunction to provide a value if the get returns null.
* @return the value if found otherwise the missing value.
*/
public long computeIfAbsent(final int key, final IntToLongFunction mappingFunction) {
long value = get(key);
if (value == missingValue) {
value = mappingFunction.applyAsLong(key);
if (value != missingValue) {
put(key, value);
}
}
return value;
}
// ---------------- Boxed Versions Below ----------------
/** {@inheritDoc} */
@Nullable
public Long get(final Object key) {
return valOrNull(get((int) key));
}
/** {@inheritDoc} */
public Long put(final Integer key, final Long value) {
return valOrNull(put((int) key, (long) value));
}
/** {@inheritDoc} */
public boolean containsKey(final Object key) {
return containsKey((int) key);
}
/** {@inheritDoc} */
public boolean containsValue(final Object value) {
return containsValue((long) value);
}
/** {@inheritDoc} */
public void putAll(final Map extends Integer, ? extends Long> map) {
for (final Map.Entry extends Integer, ? extends Long> entry : map.entrySet()) {
put(entry.getKey(), entry.getValue());
}
}
/** {@inheritDoc} */
public KeySet keySet() {
if (null == keySet) {
keySet = new KeySet();
}
return keySet;
}
/** {@inheritDoc} */
public ValueCollection values() {
if (null == values) {
values = new ValueCollection();
}
return values;
}
/** {@inheritDoc} */
public EntrySet entrySet() {
if (null == entrySet) {
entrySet = new EntrySet();
}
return entrySet;
}
/** {@inheritDoc} */
@Nullable
public Long remove(final Object key) {
return valOrNull(remove((int) key));
}
/**
* Remove value from the map using given key avoiding boxing.
*
* @param key whose mapping is to be removed from the map.
* @return removed value or {@link #missingValue()} if key was not found in the map.
*/
public long remove(final int key) {
final int mask = entries.length - 1;
int keyIndex = evenHash(key, mask);
long oldValue = missingValue;
while (entries[keyIndex + 1] != missingValue) {
if (entries[keyIndex] == key) {
oldValue = entries[keyIndex + 1];
entries[keyIndex + 1] = missingValue;
size--;
compactChain(keyIndex);
break;
}
keyIndex = next(keyIndex, mask);
}
return oldValue;
}
@SuppressWarnings("FinalParameters")
private void compactChain(int deleteKeyIndex) {
final int mask = entries.length - 1;
int keyIndex = deleteKeyIndex;
while (true) {
keyIndex = next(keyIndex, mask);
if (entries[keyIndex + 1] == missingValue) {
break;
}
final int hash = evenHash((int) entries[keyIndex], mask);
if ((keyIndex < hash && (hash <= deleteKeyIndex || deleteKeyIndex <= keyIndex))
|| (hash <= deleteKeyIndex && deleteKeyIndex <= keyIndex)) {
entries[deleteKeyIndex] = entries[keyIndex];
entries[deleteKeyIndex + 1] = entries[keyIndex + 1];
entries[keyIndex + 1] = missingValue;
deleteKeyIndex = keyIndex;
}
}
}
/**
* Get the minimum value stored in the map. If the map is empty then it will return {@link
* #missingValue()}
*
* @return the minimum value stored in the map.
*/
public long minValue() {
final long missingValue = this.missingValue;
long min = size == 0 ? missingValue : Long.MAX_VALUE;
final int length = entries.length;
for (int valueIndex = 1; valueIndex < length; valueIndex += 2) {
final long value = entries[valueIndex];
if (value != missingValue) {
min = Math.min(min, value);
}
}
return min;
}
/**
* Get the maximum value stored in the map. If the map is empty then it will return {@link
* #missingValue()}
*
* @return the maximum value stored in the map.
*/
public long maxValue() {
final long missingValue = this.missingValue;
long max = size == 0 ? missingValue : Long.MIN_VALUE;
final int length = entries.length;
for (int valueIndex = 1; valueIndex < length; valueIndex += 2) {
final long value = entries[valueIndex];
if (value != missingValue) {
max = Math.max(max, value);
}
}
return max;
}
/** {@inheritDoc} */
public String toString() {
if (isEmpty()) {
return "{}";
}
final EntryIterator entryIterator = new EntryIterator();
entryIterator.reset();
final StringBuilder sb = new StringBuilder().append('{');
while (true) {
entryIterator.next();
sb.append(entryIterator.getIntKey()).append('=').append(entryIterator.getLongValue());
if (!entryIterator.hasNext()) {
return sb.append('}').toString();
}
sb.append(',').append(' ');
}
}
/**
* Primitive specialised version of {@link #replace(Object, Object)}
*
* @param key key with which the specified value is associated
* @param value value to be associated with the specified key
* @return the previous value associated with the specified key, or {@link #missingValue()} if
* there was no mapping for the key.
*/
public long replace(final int key, final long value) {
long currentValue = get(key);
if (currentValue != missingValue) {
currentValue = put(key, value);
}
return currentValue;
}
/**
* Primitive specialised version of {@link #replace(Object, Object, Object)}
*
* @param key key with which the specified value is associated
* @param oldValue value expected to be associated with the specified key
* @param newValue value to be associated with the specified key
* @return {@code true} if the value was replaced
*/
public boolean replace(final int key, final long oldValue, final long newValue) {
final long curValue = get(key);
if (curValue != oldValue || curValue == missingValue) {
return false;
}
put(key, newValue);
return true;
}
/** {@inheritDoc} */
public boolean equals(final Object o) {
if (this == o) {
return true;
}
if (!(o instanceof Map)) {
return false;
}
final Map, ?> that = (Map, ?>) o;
return size == that.size() && entrySet().equals(that.entrySet());
}
public int hashCode() {
return entrySet().hashCode();
}
private static int next(final int index, final int mask) {
return (index + 2) & mask;
}
private void capacity(final int newCapacity) {
final int entriesLength = newCapacity * 2;
if (entriesLength < 0) {
throw new IllegalStateException("max capacity reached at size=" + size);
}
/*@DoNotSub*/ resizeThreshold = (int) (newCapacity * loadFactor);
entries = new long[entriesLength];
Arrays.fill(entries, missingValue);
}
@Nullable
private Long valOrNull(final long value) {
return value == missingValue ? null : value;
}
// ---------------- Utility Classes ----------------
/** Base iterator implementation. */
abstract class AbstractIterator implements Serializable {
private static final long serialVersionUID = 5262459454112462433L;
/** Is current position valid. */
protected boolean isPositionValid = false;
private int remaining;
private int positionCounter;
private int stopCounter;
final void reset() {
isPositionValid = false;
remaining = Int2LongHashMap.this.size;
final long missingValue = Int2LongHashMap.this.missingValue;
final long[] entries = Int2LongHashMap.this.entries;
final int capacity = entries.length;
int keyIndex = capacity;
if (entries[capacity - 1] != missingValue) {
for (int i = 1; i < capacity; i += 2) {
if (entries[i] == missingValue) {
keyIndex = i - 1;
break;
}
}
}
stopCounter = keyIndex;
positionCounter = keyIndex + capacity;
}
/**
* Returns position of the key of the current entry.
*
* @return key position.
*/
protected final int keyPosition() {
return positionCounter & entries.length - 1;
}
/**
* Number of remaining elements.
*
* @return number of remaining elements.
*/
public int remaining() {
return remaining;
}
/**
* Check if there are more elements remaining.
*
* @return {@code true} if {@code remaining > 0}.
*/
public boolean hasNext() {
return remaining > 0;
}
/**
* Advance to the next entry.
*
* @throws NoSuchElementException if no more entries available.
*/
protected final void findNext() {
if (!hasNext()) {
throw new NoSuchElementException();
}
final long[] entries = Int2LongHashMap.this.entries;
final long missingValue = Int2LongHashMap.this.missingValue;
final int mask = entries.length - 1;
for (int keyIndex = positionCounter - 2; keyIndex >= stopCounter; keyIndex -= 2) {
final int index = keyIndex & mask;
if (entries[index + 1] != missingValue) {
isPositionValid = true;
positionCounter = keyIndex;
--remaining;
return;
}
}
isPositionValid = false;
throw new IllegalStateException();
}
/** {@inheritDoc} */
public void remove() {
if (isPositionValid) {
final int position = keyPosition();
entries[position + 1] = missingValue;
--size;
compactChain(position);
isPositionValid = false;
} else {
throw new IllegalStateException();
}
}
}
/** Iterator over keys which supports access to unboxed keys via {@link #nextValue()}. */
public final class KeyIterator extends AbstractIterator
implements Iterator, Serializable {
private static final long serialVersionUID = 9151493609653852972L;
public Integer next() {
return nextValue();
}
/**
* Return next key.
*
* @return next key.
*/
public int nextValue() {
findNext();
return (int) entries[keyPosition()];
}
}
/** Iterator over values which supports access to unboxed values. */
public final class ValueIterator extends AbstractIterator
implements Iterator, Serializable {
private static final long serialVersionUID = -5670291734793552927L;
public Long next() {
return nextValue();
}
/**
* Return next value.
*
* @return next value.
*/
public long nextValue() {
findNext();
return entries[keyPosition() + 1];
}
}
/** Iterator over entries which supports access to unboxed keys and values. */
public final class EntryIterator extends AbstractIterator
implements Iterator>, Entry, Serializable {
private static final long serialVersionUID = 1744408438593481051L;
public Integer getKey() {
return getIntKey();
}
/**
* Returns the key of the current entry.
*
* @return the key.
*/
public int getIntKey() {
return (int) entries[keyPosition()];
}
public Long getValue() {
return getLongValue();
}
/**
* Returns the value of the current entry.
*
* @return the value.
*/
public long getLongValue() {
return entries[keyPosition() + 1];
}
public Long setValue(final Long value) {
return setValue(value.longValue());
}
/**
* Sets the value of the current entry.
*
* @param value to be set.
* @return previous value of the entry.
*/
public long setValue(final long value) {
if (!isPositionValid) {
throw new IllegalStateException();
}
if (missingValue == value) {
throw new IllegalArgumentException();
}
final int keyPosition = keyPosition();
final long prevValue = entries[keyPosition + 1];
entries[keyPosition + 1] = value;
return prevValue;
}
public Entry next() {
findNext();
if (shouldAvoidAllocation) {
return this;
}
return allocateDuplicateEntry();
}
private Entry allocateDuplicateEntry() {
return new MapEntry(getIntKey(), getLongValue());
}
/** {@inheritDoc} */
public int hashCode() {
return Integer.hashCode(getIntKey()) ^ Long.hashCode(getLongValue());
}
/** {@inheritDoc} */
public boolean equals(final Object o) {
if (this == o) {
return true;
}
if (!(o instanceof Entry)) {
return false;
}
final Entry, ?> that = (Entry, ?>) o;
return Objects.equals(getKey(), that.getKey()) && Objects.equals(getValue(), that.getValue());
}
/** An {@link java.util.Map.Entry} implementation. */
public final class MapEntry implements Entry {
private final int k;
private final long v;
/**
* Constructs entry with given key and value.
*
* @param k key.
* @param v value.
*/
public MapEntry(final int k, final long v) {
this.k = k;
this.v = v;
}
public Integer getKey() {
return k;
}
public Long getValue() {
return v;
}
public Long setValue(final Long value) {
return Int2LongHashMap.this.put(k, value.longValue());
}
public int hashCode() {
return Integer.hashCode(getIntKey()) ^ Long.hashCode(getLongValue());
}
public boolean equals(final Object o) {
if (!(o instanceof Map.Entry)) {
return false;
}
final Entry, ?> e = (Entry, ?>) o;
return (e.getKey() != null && e.getValue() != null)
&& (e.getKey().equals(k) && e.getValue().equals(v));
}
public String toString() {
return k + "=" + v;
}
}
}
/** Set of keys which supports optional cached iterators to avoid allocation. */
public final class KeySet extends AbstractSet implements Serializable {
private static final long serialVersionUID = -7645453993079742625L;
private final KeyIterator keyIterator = shouldAvoidAllocation ? new KeyIterator() : null;
/** {@inheritDoc} */
public KeyIterator iterator() {
KeyIterator keyIterator = this.keyIterator;
if (null == keyIterator) {
keyIterator = new KeyIterator();
}
keyIterator.reset();
return keyIterator;
}
/** {@inheritDoc} */
public int size() {
return Int2LongHashMap.this.size();
}
/** {@inheritDoc} */
public boolean isEmpty() {
return Int2LongHashMap.this.isEmpty();
}
/** {@inheritDoc} */
public void clear() {
Int2LongHashMap.this.clear();
}
/** {@inheritDoc} */
public boolean contains(final Object o) {
return contains((int) o);
}
/**
* Checks if key is contained in the map without boxing.
*
* @param key to check.
* @return {@code true} if key is contained in this map.
*/
public boolean contains(final int key) {
return containsKey(key);
}
}
/** Collection of values which supports optionally cached iterators to avoid allocation. */
public final class ValueCollection extends AbstractCollection implements Serializable {
private static final long serialVersionUID = -8925598924781601919L;
private final ValueIterator valueIterator = shouldAvoidAllocation ? new ValueIterator() : null;
/** {@inheritDoc} */
public ValueIterator iterator() {
ValueIterator valueIterator = this.valueIterator;
if (null == valueIterator) {
valueIterator = new ValueIterator();
}
valueIterator.reset();
return valueIterator;
}
/** {@inheritDoc} */
public int size() {
return Int2LongHashMap.this.size();
}
/** {@inheritDoc} */
public boolean contains(final Object o) {
return contains((long) o);
}
/**
* Checks if the value is contained in the map.
*
* @param value to be checked.
* @return {@code true} if value is contained in this map.
*/
public boolean contains(final long value) {
return containsValue(value);
}
}
/** Set of entries which supports optionally cached iterators to avoid allocation. */
public final class EntrySet extends AbstractSet>
implements Serializable {
private static final long serialVersionUID = 63641283589916174L;
private final EntryIterator entryIterator = shouldAvoidAllocation ? new EntryIterator() : null;
/** {@inheritDoc} */
public EntryIterator iterator() {
EntryIterator entryIterator = this.entryIterator;
if (null == entryIterator) {
entryIterator = new EntryIterator();
}
entryIterator.reset();
return entryIterator;
}
/** {@inheritDoc} */
public int size() {
return Int2LongHashMap.this.size();
}
/** {@inheritDoc} */
public boolean isEmpty() {
return Int2LongHashMap.this.isEmpty();
}
/** {@inheritDoc} */
public void clear() {
Int2LongHashMap.this.clear();
}
/** {@inheritDoc} */
public boolean contains(final Object o) {
if (!(o instanceof Entry)) {
return false;
}
final Entry, ?> entry = (Entry, ?>) o;
final Long value = get(entry.getKey());
return value != null && value.equals(entry.getValue());
}
/** {@inheritDoc} */
public Object[] toArray() {
return toArray(new Object[size()]);
}
/** {@inheritDoc} */
@SuppressWarnings("unchecked")
public T[] toArray(final T[] a) {
final T[] array =
a.length >= size
? a
: (T[]) java.lang.reflect.Array.newInstance(a.getClass().getComponentType(), size);
final EntryIterator it = iterator();
for (int i = 0; i < array.length; i++) {
if (it.hasNext()) {
it.next();
array[i] = (T) it.allocateDuplicateEntry();
} else {
array[i] = null;
break;
}
}
return array;
}
}
private static int evenHash(final int value, final int mask) {
final int hash = (value << 1) - (value << 8);
return hash & mask;
}
private static void validateLoadFactor(final float loadFactor) {
if (loadFactor < 0.1f || loadFactor > 0.9f) {
throw new IllegalArgumentException(
"load factor must be in the range of 0.1 to 0.9: " + loadFactor);
}
}
private static int findNextPositivePowerOfTwo(final int value) {
return 1 << (Integer.SIZE - Integer.numberOfLeadingZeros(value - 1));
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy