-
Notifications
You must be signed in to change notification settings - Fork 12
/
Copy pathabwd.ml
63 lines (49 loc) · 2.43 KB
/
abwd.ml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
(* Backwards association lists. Use in place of Map when the order of entries also matters. This module should not be opened, but used qualified. *)
open Bwd
type ('k, 'a) t = ('k * 'a) Bwd.t
let empty : ('k, 'a) t = Emp
let map (f : 'a -> 'b) (map : ('k, 'a) t) : ('k, 'b) t =
Mbwd.mmap (fun [ (x, a) ] -> (x, f a)) [ map ]
let mapi (f : 'k -> 'a -> 'b) (map : ('k, 'a) t) : ('k, 'b) t =
Mbwd.mmap (fun [ (x, a) ] -> (x, f x a)) [ map ]
let find_opt (x : 'k) (map : ('k, 'a) t) : 'a option =
Option.map snd (Bwd.find_opt (fun (y, _) -> x = y) map)
let iter (f : 'k -> 'a -> unit) (map : ('k, 'a) t) : unit =
Mbwd.miter (fun [ (k, a) ] -> f k a) [ map ]
let mem (x : 'k) (map : ('k, 'a) t) : bool = Option.is_some (find_opt x map)
let add (x : 'k) (a : 'a) (map : ('k, 'a) t) = Snoc (map, (x, a))
let fold (f : 'k -> 'a -> 'acc -> 'acc) (map : ('k, 'a) t) (start : 'acc) =
Bwd.fold_left (fun acc (x, a) -> f x a acc) start map
(* If there is a matching value, delete it and return it. *)
let rec extract (x : 'k) (map : ('k, 'a) t) : ('k, 'a) t * 'a option =
match map with
| Emp -> (Emp, None)
| Snoc (map, (y, v)) when x = y -> (map, Some v)
| Snoc (map, yv) ->
let map, xv = extract x map in
(Snoc (map, yv), xv)
exception No_such_key
(* This is like Map.update: the function is applied to (Some v) if (x,v) is in the map and to None otherwise, and if its return value is (Some w) that value replaces the existing one or becomes a new one while if its return value is None the value is deleted. However, we make sure that in the case of None -> Some, the new element is added at the *end*. *)
let update (x : 'k) (f : 'a option -> 'a option) (map : ('k, 'a) t) : ('k, 'a) t =
let rec go map =
match map with
| Emp -> raise No_such_key
| Snoc (map, (y, v)) -> (
if x <> y then Snoc (go map, (y, v))
else
match f (Some v) with
| Some w -> Snoc (map, (x, w))
| None -> map) in
try go map
with No_such_key -> (
match f None with
| Some v -> Snoc (map, (x, v))
| None -> map)
let rec find_opt_and_update_key (oldkey : 'k) (newkey : 'k) (map : ('k, 'a) t) =
match map with
| Emp -> None
| Snoc (map, (x, y)) -> (
match find_opt_and_update_key oldkey newkey map with
| Some (v, newmap) -> Some (v, Snoc (newmap, (x, y)))
| None -> if x = oldkey then Some (y, Snoc (map, (newkey, y))) else None)
let bindings (map : ('k, 'a) t) = Bwd.to_list map