mirror of
https://github.com/Minestom/Minestom.git
synced 2024-11-12 13:44:28 +01:00
Avoid slow synchronized path during reference building
This commit is contained in:
parent
fdd3e2c53c
commit
c009ef0505
@ -8,7 +8,8 @@ import java.util.*;
|
||||
import java.util.concurrent.atomic.AtomicReference;
|
||||
|
||||
final class SnapshotUpdaterImpl implements SnapshotUpdater {
|
||||
private final Map<Snapshotable, AtomicReference<Snapshot>> referenceMap = new IdentityHashMap<>();
|
||||
private final IdentityHashMap<Snapshotable, AtomicReference<Snapshot>> referenceMap = new IdentityHashMap<>();
|
||||
private IdentityHashMap<Snapshotable, AtomicReference<Snapshot>> readOnlyReferenceMap;
|
||||
private List<Entry> queue = new ArrayList<>();
|
||||
|
||||
static <T extends Snapshot> @NotNull T update(@NotNull Snapshotable snapshotable) {
|
||||
@ -19,13 +20,22 @@ final class SnapshotUpdaterImpl implements SnapshotUpdater {
|
||||
}
|
||||
|
||||
@Override
|
||||
public synchronized <T extends Snapshot> @NotNull AtomicReference<T> reference(@NotNull Snapshotable snapshotable) {
|
||||
AtomicReference<Snapshot> ref = new AtomicReference<>();
|
||||
public <T extends Snapshot> @NotNull AtomicReference<T> reference(@NotNull Snapshotable snapshotable) {
|
||||
AtomicReference<Snapshot> ref;
|
||||
// Very often the same snapshotable is referenced multiple times.
|
||||
var readOnly = this.readOnlyReferenceMap;
|
||||
if (readOnly != null && (ref = readOnly.get(snapshotable)) != null) {
|
||||
return (AtomicReference<T>) ref;
|
||||
}
|
||||
// If this is a new snapshotable, we need to create a new reference.
|
||||
synchronized (this) {
|
||||
ref = new AtomicReference<>();
|
||||
var prev = referenceMap.putIfAbsent(snapshotable, ref);
|
||||
if (prev != null) return (AtomicReference<T>) prev;
|
||||
this.queue.add(new Entry(snapshotable, ref));
|
||||
return (AtomicReference<T>) ref;
|
||||
}
|
||||
}
|
||||
|
||||
record Entry(Snapshotable snapshotable, AtomicReference<Snapshot> ref) {
|
||||
}
|
||||
@ -34,6 +44,7 @@ final class SnapshotUpdaterImpl implements SnapshotUpdater {
|
||||
List<Entry> temp;
|
||||
while (!(temp = new ArrayList<>(queue)).isEmpty()) {
|
||||
queue = new ArrayList<>();
|
||||
readOnlyReferenceMap = (IdentityHashMap<Snapshotable, AtomicReference<Snapshot>>) referenceMap.clone();
|
||||
temp.parallelStream().forEach(entry -> {
|
||||
Snapshotable snap = entry.snapshotable;
|
||||
entry.ref.set(Objects.requireNonNull(snap.updateSnapshot(this), "Snapshot must not be null after an update!"));
|
||||
|
Loading…
Reference in New Issue
Block a user