From 91e6ab4fcf65bcd92938c89fe3a91348cd8452f7 Mon Sep 17 00:00:00 2001 From: Xiaoyao Date: Tue, 27 Sep 2016 19:13:06 -0500 Subject: [PATCH] LRU cache guarantee to keep size under limit (#3510) * LRU cache guarantee to keep size under limit * address comments * fix failed tests in jdk7 --- .../client/cache/ByteCountingLRUMap.java | 54 +++++++++++-------- .../client/cache/ByteCountingLRUMapTest.java | 8 +-- 2 files changed, 35 insertions(+), 27 deletions(-) diff --git a/server/src/main/java/io/druid/client/cache/ByteCountingLRUMap.java b/server/src/main/java/io/druid/client/cache/ByteCountingLRUMap.java index 1f4b1ce6211..c0677b1602b 100644 --- a/server/src/main/java/io/druid/client/cache/ByteCountingLRUMap.java +++ b/server/src/main/java/io/druid/client/cache/ByteCountingLRUMap.java @@ -22,10 +22,14 @@ package io.druid.client.cache; import com.metamx.common.logger.Logger; import java.nio.ByteBuffer; +import java.util.ArrayList; import java.util.Collections; +import java.util.Iterator; import java.util.LinkedHashMap; +import java.util.List; import java.util.Map; import java.util.Set; +import java.util.concurrent.atomic.AtomicLong; /** */ @@ -37,8 +41,8 @@ class ByteCountingLRUMap extends LinkedHashMap private final int logEvictionCount; private final long sizeInBytes; - private volatile long numBytes; - private volatile long evictionCount; + private final AtomicLong numBytes; + private final AtomicLong evictionCount; public ByteCountingLRUMap( final long sizeInBytes @@ -58,46 +62,49 @@ class ByteCountingLRUMap extends LinkedHashMap this.sizeInBytes = sizeInBytes; logEvictions = logEvictionCount != 0; - numBytes = 0; - evictionCount = 0; + numBytes = new AtomicLong(0L); + evictionCount = new AtomicLong(0L); } public long getNumBytes() { - return numBytes; + return numBytes.get(); } public long getEvictionCount() { - return evictionCount; + return evictionCount.get(); } @Override public byte[] put(ByteBuffer key, byte[] value) { - numBytes += key.remaining() + value.length; - return super.put(key, value); - } - - @Override - protected boolean removeEldestEntry(Map.Entry eldest) - { - if (numBytes > sizeInBytes) { - ++evictionCount; - if (logEvictions && evictionCount % logEvictionCount == 0) { + numBytes.addAndGet(key.remaining() + value.length); + Iterator> it = entrySet().iterator(); + List keysToRemove = new ArrayList<>(); + long totalEvictionSize = 0L; + while (numBytes.get() - totalEvictionSize > sizeInBytes && it.hasNext()) { + evictionCount.incrementAndGet(); + if (logEvictions && evictionCount.get() % logEvictionCount == 0) { log.info( "Evicting %,dth element. Size[%,d], numBytes[%,d], averageSize[%,d]", evictionCount, size(), - numBytes, - numBytes / size() + numBytes.get(), + numBytes.get() / size() ); } - numBytes -= eldest.getKey().remaining() + eldest.getValue().length; - return true; + Map.Entry next = it.next(); + totalEvictionSize += next.getKey().remaining() + next.getValue().length; + keysToRemove.add(next.getKey()); } - return false; + + for (ByteBuffer keyToRemove : keysToRemove) { + remove(keyToRemove); + } + + return super.put(key, value); } @Override @@ -105,7 +112,8 @@ class ByteCountingLRUMap extends LinkedHashMap { byte[] value = super.remove(key); if(value != null) { - numBytes -= ((ByteBuffer)key).remaining() + value.length; + long delta = -((ByteBuffer)key).remaining() - value.length; + numBytes.addAndGet(delta); } return value; } @@ -123,7 +131,7 @@ class ByteCountingLRUMap extends LinkedHashMap @Override public void clear() { - numBytes = 0; + numBytes.set(0L); super.clear(); } } diff --git a/server/src/test/java/io/druid/client/cache/ByteCountingLRUMapTest.java b/server/src/test/java/io/druid/client/cache/ByteCountingLRUMapTest.java index 9fce4868f76..a02c8c547de 100644 --- a/server/src/test/java/io/druid/client/cache/ByteCountingLRUMapTest.java +++ b/server/src/test/java/io/druid/client/cache/ByteCountingLRUMapTest.java @@ -65,8 +65,8 @@ public class ByteCountingLRUMapTest Assert.assertEquals(ByteBuffer.wrap(eightyEightVal), ByteBuffer.wrap(map.get(tenKey))); map.put(twoByte, oneByte.array()); - assertMapValues(2, 101, 2); - Assert.assertEquals(ByteBuffer.wrap(eightyEightVal), ByteBuffer.wrap(map.get(tenKey))); + assertMapValues(1, 3, 3); + Assert.assertEquals(null, map.get(tenKey)); Assert.assertEquals(oneByte, ByteBuffer.wrap(map.get(twoByte))); Iterator it = map.keySet().iterator(); @@ -80,10 +80,10 @@ public class ByteCountingLRUMapTest for(ByteBuffer buf : toRemove) { map.remove(buf); } - assertMapValues(1, 3, 2); + assertMapValues(1, 3, 3); map.remove(twoByte); - assertMapValues(0, 0, 2); + assertMapValues(0, 0, 3); } private void assertMapValues(final int size, final int numBytes, final int evictionCount)