Description
According to javadoc:
This class is usually preferable to AtomicLong when multiple threads update a common sum that is used for purposes such as collecting statistics, not for fine-grained synchronization control. Under low update contention, the two classes have similar characteristics. But under high contention, expected throughput of this class is significantly higher, at the expense of higher space consumption.
I tried to update the StatisticsCollector
to use LongAdder
but it is not really possible to make it very efficient,
because StatisticsCollector
uses increment and get pattern.
Example:
@Override
public <K> long incrementLoadCount(IncrementLoadCountStatisticsContext<K> context) {
return loadCount.incrementAndGet();
}
With LongAdder
it would be something like:
@Override
public <K> long incrementLoadCount(IncrementLoadCountStatisticsContext<K> context) {
loadCount.increment();
return loadCount.sum(); // this could be more expensive
}
sum()
is triggered on every call, so it could slow down a bit, but luckily the return value is never used, at least internally.
I'm aware that changing all methods to return void it is a breaking change but in theory it could greatly improve performance under high contention (i.e. lot of threads updating statistics).
In case you're interested, here is the implementation:
/**
* This simple collector uses {@link java.util.concurrent.atomic.LongAdder}s to collect
* statistics
*
* @see org.dataloader.stats.StatisticsCollector
*/
public class ScalableStatisticsCollector implements StatisticsCollector {
private final LongAdder loadCount = new LongAdder();
private final LongAdder batchInvokeCount = new LongAdder();
private final LongAdder batchLoadCount = new LongAdder();
private final LongAdder cacheHitCount = new LongAdder();
private final LongAdder batchLoadExceptionCount = new LongAdder();
private final LongAdder loadErrorCount = new LongAdder();
@Override
public <K> long incrementLoadCount(IncrementLoadCountStatisticsContext<K> context) {
loadCount.increment();
return loadCount.sum();
}
@Deprecated
@Override
public long incrementLoadCount() {
return incrementLoadCount(null);
}
@Override
public <K> long incrementLoadErrorCount(IncrementLoadErrorCountStatisticsContext<K> context) {
loadErrorCount.increment();
return loadErrorCount.sum();
}
@Deprecated
@Override
public long incrementLoadErrorCount() {
return incrementLoadErrorCount(null);
}
@Override
public <K> long incrementBatchLoadCountBy(long delta, IncrementBatchLoadCountByStatisticsContext<K> context) {
batchInvokeCount.increment();
batchLoadCount.add(delta);
return batchLoadCount.sum();
}
@Deprecated
@Override
public long incrementBatchLoadCountBy(long delta) {
return incrementBatchLoadCountBy(delta, null);
}
@Override
public <K> long incrementBatchLoadExceptionCount(IncrementBatchLoadExceptionCountStatisticsContext<K> context) {
batchLoadExceptionCount.increment();
return batchLoadExceptionCount.sum();
}
@Deprecated
@Override
public long incrementBatchLoadExceptionCount() {
return incrementBatchLoadExceptionCount(null);
}
@Override
public <K> long incrementCacheHitCount(IncrementCacheHitCountStatisticsContext<K> context) {
cacheHitCount.increment();
return cacheHitCount.sum();
}
@Deprecated
@Override
public long incrementCacheHitCount() {
return incrementCacheHitCount(null);
}
@Override
public Statistics getStatistics() {
return new Statistics(loadCount.sum(), loadErrorCount.sum(), batchInvokeCount.sum(), batchLoadCount.sum(), batchLoadExceptionCount.sum(), cacheHitCount.sum());
}
@Override
public String toString() {
return getStatistics().toString();
}
}