Avoid slow synchronized path during reference building

This commit is contained in:
themode 2022-05-04 18:24:15 +02:00
parent fdd3e2c53c
commit c009ef0505
1 changed files with 18 additions and 7 deletions

View File

@ -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,12 +20,21 @@ final class SnapshotUpdaterImpl implements SnapshotUpdater {
}
@Override
public synchronized <T extends Snapshot> @NotNull AtomicReference<T> reference(@NotNull Snapshotable snapshotable) {
AtomicReference<Snapshot> 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;
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!"));