mirror of
https://github.com/c-cube/ocaml-containers.git
synced 2025-12-06 03:05:28 -05:00
133 lines
4.6 KiB
OCaml
133 lines
4.6 KiB
OCaml
(*
|
|
Copyright (c) 2013, Simon Cruanes
|
|
All rights reserved.
|
|
|
|
Redistribution and use in source and binary forms, with or without
|
|
modification, are permitted provided that the following conditions are met:
|
|
|
|
Redistributions of source code must retain the above copyright notice, this
|
|
list of conditions and the following disclaimer. Redistributions in binary
|
|
form must reproduce the above copyright notice, this list of conditions and the
|
|
following disclaimer in the documentation and/or other materials provided with
|
|
the distribution.
|
|
|
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
|
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
|
|
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
|
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
|
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
|
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*)
|
|
|
|
(** {1 Memoization caches} *)
|
|
|
|
type 'a equal = 'a -> 'a -> bool
|
|
type 'a hash = 'a -> int
|
|
|
|
(** {2 Value interface}
|
|
|
|
Typical use case: one wants to memoize a function [f : 'a -> 'b]. Code sample:
|
|
{[
|
|
let f x =
|
|
print_endline "call f";
|
|
x + 1;;
|
|
|
|
let f' = with_cache (lru 256) f;;
|
|
f' 0;; (* prints *)
|
|
f' 1;; (* prints *)
|
|
f' 0;; (* doesn't print, returns cached value *)
|
|
]}
|
|
|
|
@since NEXT_RELEASE *)
|
|
|
|
type ('a, 'b) t
|
|
|
|
val clear : (_,_) t -> unit
|
|
(** Clear the content of the cache *)
|
|
|
|
val with_cache : ('a, 'b) t -> ('a -> 'b) -> 'a -> 'b
|
|
(** [with_cache c f] behaves like [f], but caches calls to [f] in the
|
|
cache [c]. It always returns the same value as
|
|
[f x], if [f x] returns, or raise the same exception.
|
|
However, [f] may not be called if [x] is in the cache. *)
|
|
|
|
val with_cache_rec : ('a,'b) t -> (('a -> 'b) -> 'a -> 'b) -> 'a -> 'b
|
|
(** [with_cache_rec c f] is a function that first, applies [f] to
|
|
some [f' = fix f], such that recursive calls to [f'] are cached in [c].
|
|
It is similar to {!with_cache} but with a function that takes as
|
|
first argument its own recursive version.
|
|
Examples (memoized Fibonacci function):
|
|
{[
|
|
let fib = with_cache_rec (lru 256)
|
|
(fun fib' n -> match n with
|
|
| 1 | 2 -> 1
|
|
| _ -> fib' (n-1) + fib' (n-2)
|
|
);;
|
|
|
|
fib 70;;
|
|
]}
|
|
*)
|
|
|
|
val dummy : ('a,'b) t
|
|
(** dummy cache, never stores any value *)
|
|
|
|
val linear : ?eq:'a equal -> int -> ('a, 'b) t
|
|
(** Linear cache with the given size. It stores key/value pairs in
|
|
an array and does linear search at every call, so it should only be used
|
|
with small size.
|
|
@param eq optional equality predicate for keys *)
|
|
|
|
val replacing : ?eq:'a equal -> ?hash:'a hash ->
|
|
int -> ('a,'b) t
|
|
(** Replacing cache of the given size. Equality and hash functions can be
|
|
parametrized. It's a hash table that handles collisions by replacing
|
|
the old value with the new (so a cache entry is evicted when another
|
|
entry with the same hash (modulo size) is added).
|
|
Never grows wider than the given size. *)
|
|
|
|
val lru : ?eq:'a equal -> ?hash:'a hash ->
|
|
int -> ('a,'b) t
|
|
(** LRU cache of the given size ("Least Recently Used": keys that have not been
|
|
used recently are deleted first). Never grows wider. *)
|
|
|
|
val unbounded : ?eq:'a equal -> ?hash:'a hash ->
|
|
int -> ('a,'b) t
|
|
(** Unbounded cache, backed by a Hash table. Will grow forever
|
|
unless {!clear} is called manually. *)
|
|
|
|
(** {2 Binary Caches}
|
|
TODO
|
|
|
|
module C2 : sig
|
|
type ('a, 'b, 'c) t
|
|
|
|
val clear : (_,_,_) t -> unit
|
|
|
|
val with_cache : ('a, 'b, 'c) t -> ('a -> 'b -> 'c) -> 'a -> 'b -> 'c
|
|
|
|
val with_cache_rec : ('a,'b,'c) t ->
|
|
(('a -> 'b -> 'c) -> 'a -> 'b -> 'c) ->
|
|
'a -> 'b -> 'c
|
|
|
|
val dummy : ('a,'b,'c) t
|
|
|
|
val linear : ?eq1:('a -> 'a -> bool) -> ?eq2:('b -> 'b -> bool) ->
|
|
int -> ('a, 'b, 'c) t
|
|
|
|
val replacing : ?eq1:('a -> 'a -> bool) -> ?hash1:('a -> int) ->
|
|
?eq2:('b -> 'b -> bool) -> ?hash2:('b -> int) ->
|
|
int -> ('a,'b,'c) t
|
|
|
|
val lru : ?eq1:('a -> 'a -> bool) -> ?hash1:('a -> int) ->
|
|
?eq2:('b -> 'b -> bool) -> ?hash2:('b -> int) ->
|
|
int -> ('a,'b,'c) t
|
|
|
|
val unbounded : ?eq1:('a -> 'a -> bool) -> ?hash1:('a -> int) ->
|
|
?eq2:('b -> 'b -> bool) -> ?hash2:('b -> int) ->
|
|
int -> ('a,'b,'c) t
|
|
end
|
|
*)
|