Memoize is Memoizer for Golang. Extremely fast, Zero dependency, Zero allocation, guarantees duplicate call supression. And safe for concurrent use in high concurrency environments.
Memoize uses Go generics to supporte any functions of type:
func[T1, T2 any](T1) T2
func[T1, T2 any](T1) (T2, error)
Memoize premetives offers the following guarantees for Memoizer
and MemoizerWithErr
:
- Duplicate function call suppression
- 0 allocation
- Dependency only on stdlib
- Low overhead
And for MemoizerWithCache*
premitives:
- Duplicate function call suppression
- Dependency only on stdlib
The reason MemoizerWithCache
doesn't guarantee 0 allocation is that it needs extra synchronization mechanism to guarantee the duplicate function call suppression which adds around 5-8 allocations/op and adds an overhead around 400-800ns/op
var NewPage = memoize.New(func(name string) (p Page) {
// your normal logic here
})
The previous snippet will make sure the function runs only once for each
value of name
and will return the same value for the same input always.
Also function that returns 2 values can be memoized:
var NewPage = memoize.NewWithErr(func(name string) (p Page, err error) {
// your normal logic here
})
Also the package offers the same interface but with the option to provide your own cache
var NewPage = memoize.NewWithCache(cache, func(name string) (p Page) {
// your normal logic here
})
var NewPage = memoize.NewWithCacheErr(cache, func(name string) (p Page, err error) {
// your normal logic here
})
where cache
implements the interface:
type Cacher[K any, V any] interface {
Store(key K, value V)
Load(key K) (value V, ok bool)
}
memoize
require the cache interface to implement two simpleLoad
andStore
functions- So you can adapt any other caching library to
memoize
- This will give you a very powerful memoization patterns where you can store your cache in memory, file, remote system or have failovers with multiple layers of caching.
- This also means
memoize
package will not remove any items from the cache, it's the cache implementation responsibility to manage it's size, TTL, and communication with remote systems memoize
comes with 1 concurrency safe implementation of the cacheCache
, stored in memory and uses Go generics. packed bysync.Map
memoize
include a subpackagecache
which provides several implementations for theCacher
interfaceCache
is a simple in-memory forever cacherWithFallback
,WithReadOnly
, andWithWriteOnly
...etc wraps a cacher or more to provide or supress functionalitycache/adapters
subpackage include adapters for popular Go caches toCacher
interface. such as Hashicorp/LRU
cache/adapters
include adapters for popular go caches. here are examples for using them with memoize
import (
"github.com/patrickmn/go-cache"
"github.com/emad-elsaid/memoize"
"github.com/emad-elsaid/memoize/cache/adapters/patrickmn"
)
// strlen function memoized for 5 minutes with expired items cleaned every 10 minutes
var strlen = memoize.NewWithCache(
patrickmn.GoCache[int](
cache.New(5*time.Minute, 10*time.Minute)
),
func(s string) int {
return len(s)
},
)
import (
"https://github.com/hashicorp/golang-lru/v2"
"github.com/emad-elsaid/memoize"
"github.com/emad-elsaid/memoize/cache/adapters/hashicorp"
)
// strlen function memoized with max 1000 stored items
var strlen = memoize.NewWithCache(
hashicorp.LRU(
lru.New[string, int](1000),
),
func(s string) int { return len(s) },
)
import (
"github.com/dgraph-io/ristretto/v2"
"github.com/emad-elsaid/memoize"
"github.com/emad-elsaid/memoize/cache/adapters/dgraphio"
)
// example from ristretto README.md
var c, _ = ristretto.NewCache(&ristretto.Config[string, int]{
NumCounters: 1e7, // number of keys to track frequency of (10M).
MaxCost: 1 << 30, // maximum cost of cache (1GB).
BufferItems: 64, // number of keys per Get buffer.
})
var strlen = memoize.NewWithCache(
dgraphio.Ristretto(c),
func(s string) int {
return len(s)
},
)
import (
"github.com/emad-elsaid/memoize"
"github.com/emad-elsaid/memoize/cache/adapters/goburrow"
"github.com/goburrow/cache"
)
var strlen = memoize.NewWithCache(
goburrow.Mango[cache.Key, cache.Value, string, int](
cache.New(
cache.WithMaximumSize(100), // Limit number of entries in the cache.
cache.WithExpireAfterAccess(1*time.Minute), // Expire entries after 1 minute since last accessed.
cache.WithRefreshAfterWrite(2*time.Minute), // Expire entries after 2 minutes since last created.
),
),
func(s string) int {
return len(s)
},
)
- https://github.com/bradfitz/gomemcache
- https://github.com/redis/go-redis
- https://github.com/redis/rueidis
- https://github.com/coocood/freecache
- https://pegasus.apache.org/
- https://github.com/hazelcast/hazelcast-go-client
- https://github.com/allegro/bigcache
Each struct is tested with two benchmarks:
- Sequencial executions
- Parallel executions
And each benchmark is repeated for different key space sizes to see how it reacts to smaller vs. large key spaces
pkg: github.com/emad-elsaid/memoize
cpu: 11th Gen Intel(R) Core(TM) i7-1185G7 @ 3.00GHz
BenchmarkMemoizer/Keys:10-8 52843860 22.98 ns/op 0 B/op 0 allocs/op
BenchmarkMemoizer/Keys:100-8 48128289 24.24 ns/op 0 B/op 0 allocs/op
BenchmarkMemoizer/Keys:1000-8 38378518 30.21 ns/op 0 B/op 0 allocs/op
BenchmarkMemoizer/Keys:10000-8 30157987 38.10 ns/op 0 B/op 0 allocs/op
BenchmarkMemoizer/Keys:100000-8 23937156 49.28 ns/op 1 B/op 0 allocs/op
BenchmarkMemoizerParallel/Keys:10-8 91730824 12.99 ns/op 0 B/op 0 allocs/op
BenchmarkMemoizerParallel/Keys:100-8 88690077 13.41 ns/op 0 B/op 0 allocs/op
BenchmarkMemoizerParallel/Keys:1000-8 70860706 14.15 ns/op 0 B/op 0 allocs/op
BenchmarkMemoizerParallel/Keys:10000-8 79582891 15.17 ns/op 0 B/op 0 allocs/op
BenchmarkMemoizerParallel/Keys:100000-8 48657999 23.51 ns/op 1 B/op 0 allocs/op
BenchmarkMemoizerWithCache/Keys:10-8 2408961 511.0 ns/op 336 B/op 8 allocs/op
BenchmarkMemoizerWithCache/Keys:100-8 2546571 454.8 ns/op 336 B/op 8 allocs/op
BenchmarkMemoizerWithCache/Keys:1000-8 1976749 560.7 ns/op 336 B/op 8 allocs/op
BenchmarkMemoizerWithCache/Keys:10000-8 2341108 491.9 ns/op 336 B/op 8 allocs/op
BenchmarkMemoizerWithCache/Keys:100000-8 2158876 549.7 ns/op 342 B/op 8 allocs/op
BenchmarkMemoizerWithCacheParallel/Keys:10-8 2487337 479.9 ns/op 313 B/op 7 allocs/op
BenchmarkMemoizerWithCacheParallel/Keys:100-8 2412850 495.6 ns/op 299 B/op 7 allocs/op
BenchmarkMemoizerWithCacheParallel/Keys:1000-8 2473747 492.4 ns/op 188 B/op 5 allocs/op
BenchmarkMemoizerWithCacheParallel/Keys:10000-8 2268688 528.3 ns/op 151 B/op 5 allocs/op
BenchmarkMemoizerWithCacheParallel/Keys:100000-8 1733913 717.1 ns/op 195 B/op 5 allocs/op
BenchmarkMemoizerWithCacheErr/Keys:10-8 2456088 514.3 ns/op 336 B/op 8 allocs/op
BenchmarkMemoizerWithCacheErr/Keys:100-8 2184484 485.3 ns/op 336 B/op 8 allocs/op
BenchmarkMemoizerWithCacheErr/Keys:1000-8 2106156 529.6 ns/op 336 B/op 8 allocs/op
BenchmarkMemoizerWithCacheErr/Keys:10000-8 2386858 503.9 ns/op 336 B/op 8 allocs/op
BenchmarkMemoizerWithCacheErr/Keys:100000-8 2040483 648.2 ns/op 343 B/op 8 allocs/op
BenchmarkMemoizerWithCacheErrParallel/Keys:10-8 1978743 590.1 ns/op 314 B/op 7 allocs/op
BenchmarkMemoizerWithCacheErrParallel/Keys:100-8 1960455 602.6 ns/op 297 B/op 7 allocs/op
BenchmarkMemoizerWithCacheErrParallel/Keys:1000-8 2046734 581.2 ns/op 182 B/op 5 allocs/op
BenchmarkMemoizerWithCacheErrParallel/Keys:10000-8 1821135 644.7 ns/op 149 B/op 5 allocs/op
BenchmarkMemoizerWithCacheErrParallel/Keys:100000-8 1314518 931.3 ns/op 203 B/op 6 allocs/op
BenchmarkMemoizerWithErr/Keys:10-8 43579936 25.06 ns/op 0 B/op 0 allocs/op
BenchmarkMemoizerWithErr/Keys:100-8 46920016 28.24 ns/op 0 B/op 0 allocs/op
BenchmarkMemoizerWithErr/Keys:1000-8 36951790 33.61 ns/op 0 B/op 0 allocs/op
BenchmarkMemoizerWithErr/Keys:10000-8 27627970 41.76 ns/op 0 B/op 0 allocs/op
BenchmarkMemoizerWithErr/Keys:100000-8 20259408 57.24 ns/op 1 B/op 0 allocs/op
BenchmarkMemoizerWithErrParallel/Keys:10-8 90412166 20.23 ns/op 0 B/op 0 allocs/op
BenchmarkMemoizerWithErrParallel/Keys:100-8 47339029 24.99 ns/op 0 B/op 0 allocs/op
BenchmarkMemoizerWithErrParallel/Keys:1000-8 43452195 25.43 ns/op 0 B/op 0 allocs/op
BenchmarkMemoizerWithErrParallel/Keys:10000-8 55278070 22.31 ns/op 0 B/op 0 allocs/op
BenchmarkMemoizerWithErrParallel/Keys:100000-8 29371357 37.95 ns/op 2 B/op 0 allocs/op
PASS
ok github.com/emad-elsaid/memoize 67.529s
Icon by Eucalyp Studio