Class CaffeineCache<K,V>
- java.lang.Object
-
- org.apache.solr.search.SolrCacheBase
-
- org.apache.solr.search.CaffeineCache<K,V>
-
- All Implemented Interfaces:
com.github.benmanes.caffeine.cache.RemovalListener<K,V>
,AutoCloseable
,org.apache.lucene.util.Accountable
,SolrInfoBean
,SolrMetricProducer
,SolrCache<K,V>
public class CaffeineCache<K,V> extends SolrCacheBase implements SolrCache<K,V>, org.apache.lucene.util.Accountable, com.github.benmanes.caffeine.cache.RemovalListener<K,V>
A SolrCache backed by the Caffeine caching library [1]. By default it uses the Window TinyLFU (W-TinyLFU) eviction policy.This cache supports either maximum size limit (the number of items) or maximum ram bytes limit, but not both. If both values are set then only maxRamMB limit is used and maximum size limit is ignored.
W-TinyLFU [2] is a near optimal policy that uses recency and frequency to determine which entry to evict in O(1) time. The estimated frequency is retained in a Count-Min Sketch and entries reside on LRU priority queues [3]. By capturing the historic frequency of an entry, the cache is able to outperform classic policies like LRU and LFU, as well as modern policies like ARC and LIRS. This policy performed particularly well in search workloads.
[1] https://github.com/ben-manes/caffeine [2] http://arxiv.org/pdf/1512.00727.pdf [3] http://highscalability.com/blog/2016/1/25/design-of-a-modern-cache.html
-
-
Nested Class Summary
-
Nested classes/interfaces inherited from class org.apache.solr.search.SolrCacheBase
SolrCacheBase.AutoWarmCountRef
-
Nested classes/interfaces inherited from interface org.apache.solr.search.SolrCache
SolrCache.State
-
Nested classes/interfaces inherited from interface org.apache.solr.core.SolrInfoBean
SolrInfoBean.Category, SolrInfoBean.Group
-
-
Field Summary
-
Fields inherited from class org.apache.solr.search.SolrCacheBase
autowarm, regenerator
-
Fields inherited from interface org.apache.solr.search.SolrCache
ASYNC_PARAM, CLEANUP_THREAD_PARAM, EVICTIONS_PARAM, HIT_RATIO_PARAM, HITS_PARAM, INITIAL_SIZE_PARAM, INSERTS_PARAM, LOOKUPS_PARAM, MAX_IDLE_TIME_PARAM, MAX_RAM_MB_PARAM, MAX_SIZE_PARAM, RAM_BYTES_USED_PARAM, SIZE_PARAM
-
-
Constructor Summary
Constructors Constructor Description CaffeineCache()
-
Method Summary
All Methods Instance Methods Concrete Methods Modifier and Type Method Description protected void
adjustMetrics(long hitsAdjust, long insertsAdjust, long lookupsAdjust)
void
clear()
:TODO: copy from Mapvoid
close()
Frees any non-memory resourcesV
computeIfAbsent(K key, IOFunction<? super K,? extends V> mappingFunction)
Get an existing element or atomically compute it if missing.V
get(K key)
:TODO: copy from MapString
getDescription()
Simple one or two line descriptionint
getMaxRamMB()
Returns maximum size limit (in MB) if set and supported, -1 otherwise.int
getMaxSize()
Returns maximum size limit (number of items) if set and supported, -1 otherwise.String
getName()
Simple common usage name, e.g.SolrMetricsContext
getSolrMetricsContext()
Implementations should return the context used inSolrMetricProducer.initializeMetrics(SolrMetricsContext, String)
to ensure proper cleanup of metrics at the end of the life-cycle of this component.Object
init(Map<String,String> args, Object persistence, CacheRegenerator regenerator)
The initialization routine.void
initializeMetrics(SolrMetricsContext parentContext, String scope)
Initialize metrics specific to this producer.boolean
isRecursionSupported()
Check if this SolrCache supports recursive calls toSolrCache.computeIfAbsent(Object, IOFunction)
.void
onRemoval(K key, V value, com.github.benmanes.caffeine.cache.RemovalCause cause)
V
put(K key, V val)
:TODO: copy from Maplong
ramBytesUsed()
V
remove(K key)
void
setMaxRamMB(int maxRamMB)
Set maximum size limit (in MB), or -1 for unlimited.void
setMaxSize(int maxSize)
Set maximum size limit (number of items), or -1 for unlimited.int
size()
:TODO: copy from MapString
toString()
void
warm(SolrIndexSearcher searcher, SolrCache<K,V> old)
Warm this cache associated withsearcher
using theold
cache object.-
Methods inherited from class org.apache.solr.search.SolrCacheBase
calcHitRatio, getAutowarmDescription, getCategory, getState, getVersion, init, isAutowarmingOn, name, setState
-
Methods inherited from class java.lang.Object
clone, equals, finalize, getClass, hashCode, notify, notifyAll, wait, wait, wait
-
Methods inherited from interface org.apache.solr.search.SolrCache
getState, initialSearcher, name, setState
-
Methods inherited from interface org.apache.solr.core.SolrInfoBean
getCategory
-
-
-
-
Method Detail
-
init
public Object init(Map<String,String> args, Object persistence, CacheRegenerator regenerator)
Description copied from interface:SolrCache
The initialization routine. Instance specific arguments are passed in theargs
map.The persistence object will exist across different lifetimes of similar caches. For example, all filter caches will share the same persistence object, sometimes at the same time (it must be thread-safe). If null is passed, then the cache implementation should create and return a new persistence object. If not null, the passed in object should be returned again.
Since it will exist across the lifetime of many caches, care should be taken to not reference any particular cache instance and prevent it from being garbage collected (no using inner classes unless they are static).
The persistence object is designed to be used as a way for statistics to accumulate across all instances of the same type of cache, however the object may be of any type desired by the cache implementation.
The
CacheRegenerator
is what the cache uses during auto-warming to regenerate an item in the new cache from an entry in the old cache.
-
onRemoval
public void onRemoval(K key, V value, com.github.benmanes.caffeine.cache.RemovalCause cause)
-
ramBytesUsed
public long ramBytesUsed()
- Specified by:
ramBytesUsed
in interfaceorg.apache.lucene.util.Accountable
-
computeIfAbsent
public V computeIfAbsent(K key, IOFunction<? super K,? extends V> mappingFunction) throws IOException
Description copied from interface:SolrCache
Get an existing element or atomically compute it if missing.- Specified by:
computeIfAbsent
in interfaceSolrCache<K,V>
- Parameters:
key
- keymappingFunction
- function to compute the element. If the function returns a null result the cache mapping will not be created. NOTE: this function must NOT attempt to modify any mappings in the cache.- Returns:
- existing or newly computed value, null if there was no existing value and it was not possible to compute a new value (in which case the new mapping won't be created).
- Throws:
IOException
- if and only if mappingFunction threw an IOException. A cache mapping will not be created in this case
-
clear
public void clear()
Description copied from interface:SolrCache
:TODO: copy from Map
-
size
public int size()
Description copied from interface:SolrCache
:TODO: copy from Map
-
close
public void close() throws IOException
Description copied from interface:SolrCache
Frees any non-memory resources- Specified by:
close
in interfaceAutoCloseable
- Specified by:
close
in interfaceSolrCache<K,V>
- Specified by:
close
in interfaceSolrMetricProducer
- Throws:
IOException
-
getMaxSize
public int getMaxSize()
Description copied from interface:SolrCache
Returns maximum size limit (number of items) if set and supported, -1 otherwise.- Specified by:
getMaxSize
in interfaceSolrCache<K,V>
-
setMaxSize
public void setMaxSize(int maxSize)
Description copied from interface:SolrCache
Set maximum size limit (number of items), or -1 for unlimited. Note: this has effect only on implementations that support it, it's a no-op otherwise- Specified by:
setMaxSize
in interfaceSolrCache<K,V>
-
getMaxRamMB
public int getMaxRamMB()
Description copied from interface:SolrCache
Returns maximum size limit (in MB) if set and supported, -1 otherwise.- Specified by:
getMaxRamMB
in interfaceSolrCache<K,V>
-
setMaxRamMB
public void setMaxRamMB(int maxRamMB)
Description copied from interface:SolrCache
Set maximum size limit (in MB), or -1 for unlimited. Note: this has effect only on implementations that support it, it's a no-op otherwise.- Specified by:
setMaxRamMB
in interfaceSolrCache<K,V>
-
adjustMetrics
protected void adjustMetrics(long hitsAdjust, long insertsAdjust, long lookupsAdjust)
-
warm
public void warm(SolrIndexSearcher searcher, SolrCache<K,V> old)
Description copied from interface:SolrCache
Warm this cache associated withsearcher
using theold
cache object.this
andold
will have the same concrete type.
-
isRecursionSupported
public boolean isRecursionSupported()
Description copied from interface:SolrCache
Check if this SolrCache supports recursive calls toSolrCache.computeIfAbsent(Object, IOFunction)
. Caches backed byConcurrentHashMap.computeIfAbsent(Object, Function)
explicitly do not support that, but other caches might.- Specified by:
isRecursionSupported
in interfaceSolrCache<K,V>
- Returns:
- whether this cache allows recursive computations
-
getName
public String getName()
Description copied from interface:SolrInfoBean
Simple common usage name, e.g. BasicQueryHandler, or fully qualified class name.- Specified by:
getName
in interfaceSolrInfoBean
-
getDescription
public String getDescription()
Description copied from interface:SolrInfoBean
Simple one or two line description- Specified by:
getDescription
in interfaceSolrInfoBean
-
getSolrMetricsContext
public SolrMetricsContext getSolrMetricsContext()
Description copied from interface:SolrMetricProducer
Implementations should return the context used inSolrMetricProducer.initializeMetrics(SolrMetricsContext, String)
to ensure proper cleanup of metrics at the end of the life-cycle of this component. This should be the child context if one was created, or null if the parent context was used.- Specified by:
getSolrMetricsContext
in interfaceSolrMetricProducer
-
initializeMetrics
public void initializeMetrics(SolrMetricsContext parentContext, String scope)
Description copied from interface:SolrMetricProducer
Initialize metrics specific to this producer.- Specified by:
initializeMetrics
in interfaceSolrMetricProducer
- Parameters:
parentContext
- parent metrics context. If this component has the same life-cycle as the parent it can simply use the parent context, otherwise it should obtain a child context usingSolrMetricsContext.getChildContext(Object)
passingthis
as the child object.scope
- component scope
-
-