Small performance improvements in ResourceLeakDetector

Motivation:

42fba015ce82ab4ab30e547c888db82fe74094e9 changed the implemention of ResourceLeakDetector to improve performance. While this was done a branch was missed that can be removed. Beside this  using a Boolean as value for the ConcurrentMap is sub-optimal as when calling remove(key, value) an uncessary instanceof check and cast is needed on each removal.

Modifications:

- Remove branch which is not needed anymore
- Replace usage of Boolean as value type of the ConcurrentMap and use our own special type which only compute hash-code one time and use a == operation for equals(...) to reduce overhead present when using Boolean.

Result:

Faster and cleaner ResourceLeakDetector.
This commit is contained in:
Norman Maurer 2016-12-01 08:13:33 +01:00 committed by Norman Maurer
parent 38b26a6145
commit cfd016cc63

View File

@ -144,7 +144,7 @@ public class ResourceLeakDetector<T> {
}
/** the collection of active resources */
private final ConcurrentMap<DefaultResourceLeak, Boolean> allLeaks = PlatformDependent.newConcurrentHashMap();
private final ConcurrentMap<DefaultResourceLeak, LeakEntry> allLeaks = PlatformDependent.newConcurrentHashMap();
private final ReferenceQueue<Object> refQueue = new ReferenceQueue<Object>();
private final ConcurrentMap<String, Boolean> reportedLeaks = PlatformDependent.newConcurrentHashMap();
@ -306,9 +306,10 @@ public class ResourceLeakDetector<T> {
private int removedRecords;
DefaultResourceLeak(Object referent) {
super(referent, referent != null? refQueue : null);
super(referent, refQueue);
assert referent != null;
if (referent != null) {
Level level = getLevel();
if (level.ordinal() >= Level.ADVANCED.ordinal()) {
creationRecord = newRecord(3);
@ -316,10 +317,7 @@ public class ResourceLeakDetector<T> {
creationRecord = null;
}
allLeaks.put(this, Boolean.TRUE);
} else {
creationRecord = null;
}
allLeaks.put(this, LeakEntry.INSTANCE);
}
@Override
@ -343,7 +341,7 @@ public class ResourceLeakDetector<T> {
@Override
public boolean close() {
// Use the ConcurrentMap remove method, which avoids allocating an iterator.
return allLeaks.remove(this, Boolean.TRUE);
return allLeaks.remove(this, LeakEntry.INSTANCE);
}
@Override
@ -426,4 +424,21 @@ public class ResourceLeakDetector<T> {
return buf.toString();
}
private static final class LeakEntry {
static final LeakEntry INSTANCE = new LeakEntry();
private static final int HASH = System.identityHashCode(INSTANCE);
private LeakEntry() { }
@Override
public int hashCode() {
return HASH;
}
@Override
public boolean equals(Object obj) {
return obj == this;
}
}
}