mirror of
https://github.com/Minestom/Minestom.git
synced 2025-01-19 22:51:41 +01:00
copy-on-write index map
This commit is contained in:
parent
043c139b91
commit
2d6368f6bd
@ -1,111 +1,39 @@
|
|||||||
package net.minestom.server.utils.collection;
|
package net.minestom.server.utils.collection;
|
||||||
|
|
||||||
import it.unimi.dsi.fastutil.HashCommon;
|
import it.unimi.dsi.fastutil.objects.Object2IntOpenHashMap;
|
||||||
import org.jetbrains.annotations.ApiStatus;
|
import org.jetbrains.annotations.ApiStatus;
|
||||||
import org.jetbrains.annotations.Contract;
|
import org.jetbrains.annotations.Contract;
|
||||||
import org.jetbrains.annotations.NotNull;
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
|
||||||
import static it.unimi.dsi.fastutil.HashCommon.arraySize;
|
|
||||||
import static it.unimi.dsi.fastutil.HashCommon.maxFill;
|
|
||||||
|
|
||||||
// Most of the code comes from fastutil's Object2IntOpenHashMap
|
|
||||||
@ApiStatus.Internal
|
@ApiStatus.Internal
|
||||||
public final class IndexMap<K> {
|
public final class IndexMap<K> {
|
||||||
private K[] key;
|
private final Object2IntOpenHashMap<K> write = new Object2IntOpenHashMap<>();
|
||||||
private int[] value;
|
private Object2IntOpenHashMap<K> read = copy();
|
||||||
private int lastIndex;
|
private int lastIndex;
|
||||||
|
|
||||||
private int maxFill, size;
|
|
||||||
private final float f = 0.75f;
|
|
||||||
|
|
||||||
public IndexMap() {
|
public IndexMap() {
|
||||||
final int n = arraySize(16, f);
|
write.defaultReturnValue(-1);
|
||||||
maxFill = maxFill(n, f);
|
|
||||||
key = (K[]) new Object[n + 1];
|
|
||||||
value = new int[n + 1];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
@Contract(pure = true)
|
@Contract(pure = true)
|
||||||
public int get(@NotNull K key) {
|
public int get(@NotNull K key) {
|
||||||
final int hash = HashCommon.mix(key.hashCode());
|
int index = read.getInt(key);
|
||||||
int index = getInt(key, hash);
|
|
||||||
if (index == -1) {
|
if (index == -1) {
|
||||||
synchronized (this) {
|
synchronized (write) {
|
||||||
index = getInt(key, hash);
|
var write = this.write;
|
||||||
if (index == -1) put(key, (index = lastIndex++), hash);
|
index = write.getInt(key);
|
||||||
|
if (index == -1) {
|
||||||
|
write.put(key, (index = lastIndex++));
|
||||||
|
read = copy();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return index;
|
return index;
|
||||||
}
|
}
|
||||||
|
|
||||||
private int getInt(final K k, int hash) {
|
private Object2IntOpenHashMap<K> copy() {
|
||||||
final K[] key = this.key;
|
var map = new Object2IntOpenHashMap<>(write);
|
||||||
final int[] value = this.value;
|
map.defaultReturnValue(-1);
|
||||||
if (key.length != value.length)
|
return map;
|
||||||
return -1; // Race condition, continue to slow path
|
|
||||||
|
|
||||||
final int mask = key.length - 2;
|
|
||||||
K curr;
|
|
||||||
int pos;
|
|
||||||
// The starting point.
|
|
||||||
if ((curr = key[pos = hash & mask]) == null) return -1;
|
|
||||||
if (k.equals(curr)) return value[pos];
|
|
||||||
// There's always an unused entry.
|
|
||||||
while (true) {
|
|
||||||
if ((curr = key[pos = (pos + 1) & mask]) == null) return -1;
|
|
||||||
if (k.equals(curr)) return value[pos];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void put(final K k, final int v, int hash) {
|
|
||||||
int pos = find(k, hash);
|
|
||||||
if (pos >= 0) {
|
|
||||||
value[pos] = v;
|
|
||||||
} else {
|
|
||||||
pos = -(pos + 1);
|
|
||||||
key[pos] = k;
|
|
||||||
value[pos] = v;
|
|
||||||
if (size++ >= maxFill) rehash(arraySize(size + 1, f));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private int find(final K k, int hash) {
|
|
||||||
final K[] key = this.key;
|
|
||||||
final int mask = key.length - 2;
|
|
||||||
|
|
||||||
K curr;
|
|
||||||
int pos;
|
|
||||||
// The starting point.
|
|
||||||
if ((curr = key[pos = hash & mask]) == null)
|
|
||||||
return -(pos + 1);
|
|
||||||
if (k.equals(curr)) return pos;
|
|
||||||
// There's always an unused entry.
|
|
||||||
while (true) {
|
|
||||||
if ((curr = key[pos = (pos + 1) & mask]) == null) return -(pos + 1);
|
|
||||||
if (k.equals(curr)) return pos;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void rehash(final int newN) {
|
|
||||||
final K[] key = this.key;
|
|
||||||
final int[] value = this.value;
|
|
||||||
final int n = key.length - 1;
|
|
||||||
|
|
||||||
final int mask = newN - 1; // Note that this is used by the hashing macro
|
|
||||||
final K[] newKey = (K[]) new Object[newN + 1];
|
|
||||||
final int[] newValue = new int[newN + 1];
|
|
||||||
|
|
||||||
int i = n, pos;
|
|
||||||
for (int j = size; j-- != 0; ) {
|
|
||||||
while (((key[--i]) == null)) ;
|
|
||||||
if (!((newKey[pos = HashCommon.mix(key[i].hashCode()) & mask]) == null))
|
|
||||||
while (!((newKey[pos = (pos + 1) & mask]) == null)) ;
|
|
||||||
newKey[pos] = key[i];
|
|
||||||
newValue[pos] = value[i];
|
|
||||||
}
|
|
||||||
newValue[newN] = value[n];
|
|
||||||
this.maxFill = maxFill(n, f);
|
|
||||||
this.key = newKey;
|
|
||||||
this.value = newValue;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user