Provides a thread-safe implementation of a Least Recently Used (LRU) Cache for Julia.
An LRU Cache is a useful associative data structure (AbstractDict in Julia) that has a
set maximum size (as measured by number of elements or a custom size measure for items).
Once that size is reached, the least recently used items are removed first. A lock ensures
that data access does not lead to race conditions.
A particular use case of this package is to implement function memoization for functions that can simultaneously be called from different threads.
See LFUDACache.jl for a similar implementation of Least Frequently Used (LFU) caches.
Install with the package manager via ]add LRUCache or
using Pkg
Pkg.add("LRUCache")LRU supports the standard AbstractDict interface. Some examples of common
operations are shown below:
Creation
lru = LRU{K, V}(, maxsize = size [, by = ...] [,finalizer = ...])Create an LRU Cache with a maximum size (number of items) specified by the required
keyword argument maxsize. Here, the size can be the number of elements (default), or the
maximal total size of the values in the dictionary, as counted by an arbitrary user
function (which should return a single value of type Int) specified with the keyword
argument by. Sensible choices would for example be by = sizeof for e.g. values which
are Arrays of bitstypes, or by = Base.summarysize for values of some arbitrary user
type.
If finalizer is set, it is called for each entry that leaves the cache, with key and
value as arguments. This is useful if the cached values contain some resource that you
want to recover.
Add an item to the cache
setindex!(lru, value, key)
lru[key] = valueLookup an item in the cache
getindex(lru, key)
lru[key]Change the maxsize
resize!(lru; maxsize = size)Here, the maximal size is specified via a required keyword argument. Remember that the
maximal size is not necessarily the same as the maximal length, if a custom function was
specified using the keyword argument by in the construction of the LRU cache.
Empty the cache
empty!(lru)To effectively use LRU as a cache, several functions from the AbstractDict interface
can be used for easy checking if an item is present, and if not quickly calculating a
default.
Returns the value stored in lru for key if present. If not, stores key => default, and returns default.
Like above, except if key is not present, stores key => default(), and
returns the result. This is intended to be used in do block syntax:
get!(lru, key) do
...
endReturns the value stored in lru for key if present. If not, returns default without
storing this value in lru. Also comes in the following form:
Returns an object holding a snapshot of information about hits, misses, current size, and total size in its properties.
The caching functions get and get! each contribute a cache hit or miss on every function call (depending on whether or not the key was found). empty! resets the counts of hits and misses to 0.
The other functions, e.g. getindex iterate, haskey, setindex!, delete!, and pop! do not contribute cache hits or misses, regardless of whether or not a key is retrieved. This is because it is not possible to use these functions for caching.
Commonly, you may have some long running function that sometimes gets called with the same parameters more than once. As such, it may benefit from caching the results.
Here's our example, long running calculation:
function foo(a::Float64, b::Float64)
sleep(100)
result = a * b
endAs this function requires more than one parameter, we need a cache from
(Float64, Float64) to Float64. A cached version is then:
const lru = LRU{Tuple{Float64, Float64}, Float64}(maxsize=10)
function cached_foo(a::Float64, b::Float64)
get!(lru, (a, b)) do
foo(a,b)
end
endIf we want to see how our cache is performing, we can call cache_info to see the number of cache hits and misses.