1 = memoization.java image:https://img.shields.io/badge/email-%40metio-brightgreen.svg?style=social&label=mail["Discuss on Google Groups", link="https://groups.google.com/forum/#!forum/metio"] image:https://img.shields.io/badge/irc-%23metio.wtf-brightgreen.svg?style=social&label=IRC["Chat on IRC", link="http://webchat.freenode.net/?channels=metio.wtf"]
2 Sebastian Hoß <http://seb.xn--ho-hia.de/[@sebhoss]>
4 :project-name: memoization.java
5 :project-group: de.xn--ho-hia.memoization
6 :coverity-project: 8732
7 :codacy-project: 0ed810b7f2514f0ea1c8e86e97c803c4
8 :jdk-api: https://docs.oracle.com/javase/8/docs/api
9 :issue: https://github.com/sebhoss/memoization.java/issues
11 :toc-placement: preamble
13 image:https://img.shields.io/badge/license-cc%20zero-000000.svg?style=flat-square["CC Zero", link="http://creativecommons.org/publicdomain/zero/1.0/"]
14 pass:[<span class="image"><a class="image" href="https://maven-badges.herokuapp.com/maven-central/de.xn--ho-hia.memoization/memoization.java"><img src="https://img.shields.io/maven-central/v/de.xn--ho-hia.memoization/memoization.java.svg?style=flat-square" alt="Maven Central"></a></span>]
15 image:https://reposs.herokuapp.com/?path={github-org}/{project-name}&style=flat-square["Repository size"]
16 image:https://www.openhub.net/p/memoization-java/widgets/project_thin_badge?format=gif["Open Hub statistics", link="https://www.openhub.net/p/memoization-java"]
18 image:https://img.shields.io/travis/{github-org}/{project-name}/master.svg?style=flat-square["Build Status", link="https://travis-ci.org/{github-org}/{project-name}"]
19 image:https://img.shields.io/coveralls/{github-org}/{project-name}/master.svg?style=flat-square["Code Coverage", link="https://coveralls.io/github/{github-org}/{project-name}"]
20 image:https://img.shields.io/coverity/scan/{coverity-project}.svg?style=flat-square["Coverity Scan Result", link="https://scan.coverity.com/projects/{github-org}-memoization-java"]
21 image:https://img.shields.io/codacy/grade/{codacy-project}.svg?style=flat-square["Codacy Code Quality", link="https://www.codacy.com/app/mail_7/memoization-java"]
22 image:https://img.shields.io/badge/forkable-yes-brightgreen.svg?style=flat-square["Can this project be forked?", link="https://basicallydan.github.io/forkability/?u={github-org}&r={project-name}"]
23 image:https://img.shields.io/maintenance/yes/2016.svg?style=flat-square["Is this thing still maintained?"]
24 image:https://img.shields.io/bountysource/team/metio/activity.svg?style=flat-square["Bounties on open tickets", link="https://www.bountysource.com/teams/metio"]
26 _Java link:https://en.wikipedia.org/wiki/Memoization[memoization] library - trade space for time_
30 * Memoize calls to `Consumer`, `Function`, `Predicate`, `Supplier` and other functional interfaces in `java.util.function`
31 * Cache values using link:https://github.com/ben-manes/caffeine[Caffeine], link:https://github.com/google/guava/wiki/CachesExplained[Guava], link:https://jcp.org/en/jsr/detail?id=107[JCache] or link:{jdk-api}/java/util/concurrent/ConcurrentMap.html[`ConcurrentMap`]
32 * Customize cache key calculation
34 .Coverage of `java.util.function`
36 | | Caffeine | Guava | JCache | ConcurrentMap
38 | link:{jdk-api}/java/util/function/BiConsumer.html[BiConsumer]
44 | link:{jdk-api}/java/util/function/BiFunction.html[BiFunction]
50 | link:{jdk-api}/java/util/function/BiPredicate.html[BiPredicate]
56 | link:{jdk-api}/java/util/function/BooleanSupplier.html[BooleanSupplier]
62 | link:{jdk-api}/java/util/function/Consumer.html[Consumer]
68 | link:{jdk-api}/java/util/function/DoubleBinaryOperator.html[DoubleBinaryOperator]
71 | link:{issue}/129[#129]
74 | link:{jdk-api}/java/util/function/DoubleConsumer.html[DoubleConsumer]
80 | link:{jdk-api}/java/util/function/DoubleFunction.html[DoubleFunction]
86 | link:{jdk-api}/java/util/function/DoublePredicate.html[DoublePredicate]
92 | link:{jdk-api}/java/util/function/DoubleSupplier.html[DoubleSupplier]
98 | link:{jdk-api}/java/util/function/DoubleToIntFunction.html[DoubleToIntFunction]
100 | link:{issue}/61[#61]
101 | link:{issue}/133[#133]
104 | link:{jdk-api}/java/util/function/DoubleToLongFunction.html[DoubleToLongFunction]
106 | link:{issue}/62[#62]
107 | link:{issue}/134[#134]
110 | link:{jdk-api}/java/util/function/DoubleUnaryOperator.html[DoubleUnaryOperator]
112 | link:{issue}/63[#63]
113 | link:{issue}/135[#135]
116 | link:{jdk-api}/java/util/function/Function.html[Function]
122 | link:{jdk-api}/java/util/function/IntBinaryOperator.html[IntBinaryOperator]
125 | link:{issue}/136[#136]
128 | link:{jdk-api}/java/util/function/IntConsumer.html[IntConsumer]
134 | link:{jdk-api}/java/util/function/IntFunction.html[IntFunction]
140 | link:{jdk-api}/java/util/function/IntPredicate.html[IntPredicate]
146 | link:{jdk-api}/java/util/function/IntSupplier.html[IntSupplier]
152 | link:{jdk-api}/java/util/function/IntToDoubleFunction.html[IntToDoubleFunction]
154 | link:{issue}/68[#68]
155 | link:{issue}/140[#140]
158 | link:{jdk-api}/java/util/function/IntToLongFunction.html[IntToLongFunction]
160 | link:{issue}/69[#69]
161 | link:{issue}/141[#141]
164 | link:{jdk-api}/java/util/function/IntUnaryOperator.html[IntUnaryOperator]
166 | link:{issue}/70[#70]
167 | link:{issue}/142[#142]
170 | link:{jdk-api}/java/util/function/LongBinaryOperator.html[LongBinaryOperator]
172 | link:{issue}/71[#71]
173 | link:{issue}/143[#143]
176 | link:{jdk-api}/java/util/function/LongConsumer.html[LongConsumer]
182 | link:{jdk-api}/java/util/function/LongFunction.html[LongFunction]
188 | link:{jdk-api}/java/util/function/LongPredicate.html[LongPredicate]
194 | link:{jdk-api}/java/util/function/LongSupplier.html[LongSupplier]
200 | link:{jdk-api}/java/util/function/LongToDoubleFunction.html[LongToDoubleFunction]
202 | link:{issue}/75[#75]
203 | link:{issue}/147[#147]
206 | link:{jdk-api}/java/util/function/LongToIntFunction.html[LongToIntFunction]
208 | link:{issue}/76[#76]
209 | link:{issue}/148[#148]
212 | link:{jdk-api}/java/util/function/LongUnaryOperator.html[LongUnaryOperator]
214 | link:{issue}/77[#77]
215 | link:{issue}/149[#149]
218 | link:{jdk-api}/java/util/function/ObjDoubleConsumer.html[ObjDoubleConsumer]
220 | link:{issue}/78[#78]
221 | link:{issue}/150[#150]
224 | link:{jdk-api}/java/util/function/ObjIntConsumer.html[ObjIntConsumer]
226 | link:{issue}/79[#79]
227 | link:{issue}/151[#151]
230 | link:{jdk-api}/java/util/function/ObjLongConsumer.html[ObjLongConsumer]
232 | link:{issue}/80[#80]
233 | link:{issue}/152[#152]
236 | link:{jdk-api}/java/util/function/Predicate.html[Predicate]
242 | link:{jdk-api}/java/util/function/Supplier.html[Supplier]
248 | link:{jdk-api}/java/util/function/ToDoubleBiFunction.html[ToDoubleBiFunction]
250 | link:{issue}/82[#82]
251 | link:{issue}/154[#154]
254 | link:{jdk-api}/java/util/function/ToIntBiFunction.html[ToIntBiFunction]
256 | link:{issue}/83[#83]
257 | link:{issue}/155[#155]
260 | link:{jdk-api}/java/util/function/ToLongBiFunction.html[ToLongBiFunction]
262 | link:{issue}/84[#84]
263 | link:{issue}/156[#156]
266 | link:{jdk-api}/java/util/function/ToDoubleFunction.html[ToDoubleFunction]
268 | link:{issue}/85[#85]
269 | link:{issue}/157[#157]
272 | link:{jdk-api}/java/util/function/ToIntFunction.html[ToIntFunction]
274 | link:{issue}/86[#86]
275 | link:{issue}/158[#158]
278 | link:{jdk-api}/java/util/function/ToLongFunction.html[ToLongFunction]
280 | link:{issue}/87[#87]
281 | link:{issue}/159[#159]
286 == Development Status
288 The `ConcurrentMap` and Caffeine based implementations cover all functional interfaces from `java.util.function`, Guava and JCache modules are currently in progress. Take a look at the link:https://github.com/sebhoss/memoization.java/issues[open tickets].
292 Memoize any of the supported types by using the static factory methods supplied by either using `CaffeineMemoize` pass:[<span class="image"><a class="image" href="https://www.javadoc.io/doc/de.xn--ho-hia.memoization/memoization-caffeine"><img src="https://www.javadoc.io/badge/de.xn--ho-hia.memoization/memoization-caffeine.svg?style=flat-square&color=blue" alt="Read JavaDocs"></a></span>], `GuavaMemoize` pass:[<span class="image"><a class="image" href="https://www.javadoc.io/doc/de.xn--ho-hia.memoization/memoization-guava"><img src="https://www.javadoc.io/badge/de.xn--ho-hia.memoization/memoization-guava.svg?style=flat-square&color=blue" alt="Read JavaDocs"></a></span>], `JCacheMemoize` pass:[<span class="image"><a class="image" href="https://www.javadoc.io/doc/de.xn--ho-hia.memoization/memoization-jcache"><img src="https://www.javadoc.io/badge/de.xn--ho-hia.memoization/memoization-jcache.svg?style=flat-square&color=blue" alt="Read JavaDocs"></a></span>] or `MapMemoize` pass:[<span class="image"><a class="image" href="https://www.javadoc.io/doc/de.xn--ho-hia.memoization/memoization-core"><img src="https://www.javadoc.io/badge/de.xn--ho-hia.memoization/memoization-core.svg?style=flat-square&color=blue" alt="Read JavaDocs"></a></span>] like this:
294 === Default cache w/ default cache keys
298 // memoize in Caffeine cache
299 Consumer<INPUT> consumer = ...;
300 Consumer<INPUT> memoizedConsumer = CaffeineMemoize.consumer(consumer);
302 // memoize in Guava cache
303 Function<INPUT, OUTPUT> function = ...;
304 Function<INPUT, OUTPUT> memoizedFunction = GuavaMemoize.function(function);
306 // memoize in JCache cache
307 Predicate<INPUT> predicate = ...;
308 Predicate<INPUT> memoizedPredicate = JCacheMemoize.predicate(predicate);
310 // memoize in ConcurrentMap
311 Supplier<OUTPUT> supplier = ...;
312 Supplier<OUTPUT> memoizedSupplier = MapMemoize.supplier(supplier);
315 === Default cache w/ custom cache keys
319 // memoize in Caffeine cache
320 Consumer<INPUT> consumer = ...;
321 Function<INPUT, KEY> keyFunction = ...;
322 Consumer<INPUT> memoizedConsumer = CaffeineMemoize.consumer(consumer, keyFunction);
324 // memoize in Guava cache
325 Function<INPUT, OUTPUT> function = ...;
326 Function<INPUT, KEY> keyFunction = ...;
327 Function<INPUT, OUTPUT> memoizedFunction = GuavaMemoize.function(function, keyFunction);
329 // memoize in JCache cache
330 Predicate<INPUT> predicate = ...;
331 Function<INPUT, KEY> keyFunction = ...;
332 Predicate<INPUT> memoizedPredicate = JCacheMemoize.predicate(predicate, keyFunction);
334 // memoize in ConcurrentMap
335 Supplier<OUTPUT> supplier = ...;
336 Supplier<KEY> keySupplier = ...;
337 Supplier<OUTPUT> memoizedSupplier = MapMemoize.supplier(supplier, keySupplier);
340 === Custom cache w/ default cache keys
344 // memoize in Caffeine cache
345 Consumer<INPUT> consumer = ...;
346 Cache<INPUT, INPUT> cache = ...; // com.github.benmanes.caffeine.cache.Cache
347 Consumer<INPUT> memoizedConsumer = CaffeineMemoize.consumer(consumer, cache);
349 // memoize in Guava cache
350 Function<INPUT, OUTPUT> function = ...;
351 Cache<INPUT, OUTPUT> cache = ...; // com.google.common.cache.Cache
352 Function<INPUT, OUTPUT> memoizedFunction = GuavaMemoize.function(function, cache);
354 // memoize in JCache cache
355 Predicate<INPUT> predicate = ...;
356 Cache<INPUT, Boolean> cache = ...; // javax.cache.Cache
357 Predicate<INPUT> memoizedPredicate = JCacheMemoize.predicate(predicate, cache);
359 // memoize in ConcurrentMap
360 Supplier<OUTPUT> supplier = ...;
361 Map<String, OUTPUT> cache = ...;
362 Supplier<OUTPUT> memoizedSupplier = MapMemoize.supplier(supplier, cache);
365 === Custom cache w/ custom cache keys
369 // memoize in Caffeine cache
370 Consumer<INPUT> consumer = ...;
371 Function<INPUT, KEY> keyFunction = ...;
372 Cache<KEY, INPUT> cache = ...; // com.github.benmanes.caffeine.cache.Cache
373 Consumer<INPUT> memoizedConsumer = CaffeineMemoize.consumer(consumer, keyFunction, cache);
375 // memoize in Guava cache
376 Function<INPUT, OUTPUT> function = ...;
377 Function<INPUT, KEY> keyFunction = ...;
378 Cache<KEY, OUTPUT> cache = ...; // com.google.common.cache.Cache
379 Function<INPUT, OUTPUT> memoizedFunction = GuavaMemoize.function(function, keyFunction, cache);
381 // memoize in JCache cache
382 Predicate<INPUT> predicate = ...;
383 Function<INPUT, KEY> keyFunction = ...;
384 Cache<KEY, Boolean> cache = ...; // javax.cache.Cache
385 Predicate<INPUT> memoizedPredicate = JCacheMemoize.predicate(predicate, keyFunction, cache);
387 // memoize in ConcurrentMap
388 Supplier<OUTPUT> supplier = ...;
389 Supplier<KEY> keySupplier = ...;
390 Map<KEY, OUTPUT> cache = ...;
391 Supplier<OUTPUT> memoizedSupplier = MapMemoize.supplier(supplier, keySupplier, cache);
396 In order to use this project, declare the following inside your POM:
398 [source, xml, subs="attributes,verbatim"]
402 <groupId>{project-group}</groupId>
403 <artifactId>memoization-core</artifactId>
404 <version>${version.memoization}</version>
407 <!-- CAFFEINE ONLY -->
409 <groupId>{project-group}</groupId>
410 <artifactId>memoization-caffeine</artifactId>
411 <version>${version.memoization}</version>
414 <groupId>com.github.ben-manes.caffeine</groupId>
415 <artifactId>caffeine</artifactId>
416 <version>${version.caffeine}</version>
418 <!-- CAFFEINE ONLY -->
422 <groupId>{project-group}</groupId>
423 <artifactId>memoization-guava</artifactId>
424 <version>${version.memoization}</version>
427 <groupId>com.google.guava</groupId>
428 <artifactId>guava</artifactId>
429 <version>${version.guava}</version>
435 <groupId>{project-group}</groupId>
436 <artifactId>memoization-jcache</artifactId>
437 <version>${version.memoization}</version>
440 <groupId>javax.cache</groupId>
441 <artifactId>cache-api</artifactId>
442 <version>${version.jcache}</version>
444 <!-- Add your JCache implementation here -->
446 <groupId>...</groupId>
447 <artifactId>...</artifactId>
448 <version>...</version>
455 Replace `${version.memoization}` with the pass:[<a href="https://search.maven.org/#search%7Cga%7C1%7Cg%3Ade.xn--ho-hia.memoization">latest release</a>]. This project follows the link:http://semver.org/[semantic versioning guidelines].
459 This project is compatible with the following Java versions:
472 * link:http://www.tek271.com/software/java/memoizer[Tek271 Memoizer]
473 * link:https://github.com/kelvinguu/gitmemoizer[GitMemoizer]
474 * link:http://docs.spring.io/spring/docs/current/spring-framework-reference/html/cache.html#cache-annotations-cacheable[Spring's `@Cacheable`]
475 * link:https://github.com/marmelo/chili#memoize[Chili's `@Memoize`]
476 * link:https://clojuredocs.org/clojure.core/memoize[Clojure's `(memoize f)`]
477 * link:http://docs.groovy-lang.org/latest/html/gapi/groovy/transform/Memoized.html[Groovy's `@Memoized`]
478 * link:https://github.com/cb372/scalacache#memoization-of-method-results[ScalaCache's `memoize`]
482 To the extent possible under law, the author(s) have dedicated all copyright
483 and related and neighboring rights to this software to the public domain
484 worldwide. This software is distributed without any warranty.
486 You should have received a copy of the CC0 Public Domain Dedication along
487 with this software. If not, see http://creativecommons.org/publicdomain/zero/1.0/.
491 * https://github.com/sebhoss/memoization.java
492 * https://bitbucket.org/sebhoss/memoization.java
493 * https://gitlab.com/sebastian.hoss/memoization.java
494 * http://v2.pikacode.com/sebhoss/memoization.java
495 * http://repo.or.cz/memoization.java.git