Imported Christian Rinderknecht's polymorphic Red-Black_Trees library from commit 98b1131281
This commit is contained in:
parent
e001154714
commit
0704676724
0
vendors/Red-Black_Trees/.PolyMapMain.tag
vendored
Normal file
0
vendors/Red-Black_Trees/.PolyMapMain.tag
vendored
Normal file
0
vendors/Red-Black_Trees/.PolySetMain.tag
vendored
Normal file
0
vendors/Red-Black_Trees/.PolySetMain.tag
vendored
Normal file
0
vendors/Red-Black_Trees/.RedBlackMain.tag
vendored
Normal file
0
vendors/Red-Black_Trees/.RedBlackMain.tag
vendored
Normal file
1
vendors/Red-Black_Trees/.links
vendored
Normal file
1
vendors/Red-Black_Trees/.links
vendored
Normal file
@ -0,0 +1 @@
|
||||
../OCaml-build/Makefile
|
21
vendors/Red-Black_Trees/LICENSE
vendored
Normal file
21
vendors/Red-Black_Trees/LICENSE
vendored
Normal file
@ -0,0 +1,21 @@
|
||||
MIT License
|
||||
|
||||
Copyright (c) 2018 Christian Rinderknecht
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
in the Software without restriction, including without limitation the rights
|
||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
copies of the Software, and to permit persons to whom the Software is
|
||||
furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice shall be included in all
|
||||
copies or substantial portions of the Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
SOFTWARE.
|
4
vendors/Red-Black_Trees/Makefile.cfg
vendored
Normal file
4
vendors/Red-Black_Trees/Makefile.cfg
vendored
Normal file
@ -0,0 +1,4 @@
|
||||
SHELL := dash
|
||||
BFLAGS := -strict-sequence -w +A-48-4
|
||||
#OCAMLC := ocamlcp
|
||||
#OCAMLOPT := ocamloptp
|
37
vendors/Red-Black_Trees/PolyMap.ml
vendored
Normal file
37
vendors/Red-Black_Trees/PolyMap.ml
vendored
Normal file
@ -0,0 +1,37 @@
|
||||
(* Polymorphic maps *)
|
||||
|
||||
module RB = RedBlack
|
||||
|
||||
type ('key, 'value) t = {
|
||||
tree : ('key * 'value) RB.t;
|
||||
cmp : 'key -> 'key -> int
|
||||
}
|
||||
|
||||
type ('key, 'value) map = ('key, 'value) t
|
||||
|
||||
let create ~cmp = {tree = RB.empty; cmp}
|
||||
|
||||
let empty = {tree = RB.empty; cmp=Pervasives.compare}
|
||||
|
||||
let is_empty map = RB.is_empty map.tree
|
||||
|
||||
let add key value map =
|
||||
let cmp (k1,_) (k2,_) = map.cmp k1 k2 in
|
||||
{map with tree = RB.add ~cmp RB.New (key, value) map.tree}
|
||||
|
||||
exception Not_found
|
||||
|
||||
let find key map =
|
||||
let cmp k1 (k2,_) = map.cmp k1 k2 in
|
||||
try snd (RB.find ~cmp key map.tree) with
|
||||
RB.Not_found -> raise Not_found
|
||||
|
||||
let find_opt key map =
|
||||
try Some (find key map) with Not_found -> None
|
||||
|
||||
let bindings map =
|
||||
RB.fold_dec (fun ~elt ~acc -> elt::acc) ~init:[] map.tree
|
||||
|
||||
let iter f map = RB.iter (fun (k,v) -> f k v) map.tree
|
||||
|
||||
let fold_inc f map = RB.fold_inc (fun ~elt:(k,v) -> f k v) map.tree
|
70
vendors/Red-Black_Trees/PolyMap.mli
vendored
Normal file
70
vendors/Red-Black_Trees/PolyMap.mli
vendored
Normal file
@ -0,0 +1,70 @@
|
||||
(* Polymorphic maps
|
||||
|
||||
This module does not provide a function to merge polymorphic
|
||||
maps. Use the functorial interface of the module [Map] of the OCaml
|
||||
standard library instead, at the cost of the polymorphism on the
|
||||
keys.
|
||||
|
||||
No deletion is provided.
|
||||
*)
|
||||
|
||||
type ('key, 'value) t
|
||||
type ('key, 'value) map = ('key, 'value) t
|
||||
|
||||
(* The value of the call [create ~cmp] is an empty map with [cmp]
|
||||
being the comparison over the (future) keys.
|
||||
|
||||
The value [empty] is identical to the value of the call [create
|
||||
~cmp:Pervasives.compare].
|
||||
*)
|
||||
|
||||
val create : cmp:('key -> 'key -> int) -> ('key, 'value) t
|
||||
|
||||
val empty : ('key, 'value) t
|
||||
|
||||
(* Emptiness *)
|
||||
|
||||
val is_empty : ('key, 'value) t -> bool
|
||||
|
||||
(* The value of the call [add key value map] is a map containing all
|
||||
the bindings of the map [map], extended by the binding of [key] to
|
||||
[value]. If there is a binding for [key] in [map], its value is
|
||||
lost (and replaced by [value]). *)
|
||||
|
||||
val add : 'key -> 'value -> ('key, 'value) t -> ('key, 'value) t
|
||||
|
||||
(* The value of the call [find key map] is the value associated to the
|
||||
[key] in the map [map]. If [key] is not bound in [map], the
|
||||
exception [Not_found] is raised. *)
|
||||
|
||||
exception Not_found
|
||||
|
||||
val find : 'key -> ('key, 'value) t -> 'value
|
||||
|
||||
(* The value of the call [find_opt key map] is [Some value] if the key
|
||||
[key] is bound to [value] in the map [map], and [None]
|
||||
otherwise. *)
|
||||
|
||||
val find_opt : 'key -> ('key, 'value) t -> 'value option
|
||||
|
||||
(* The value of the call [bindings map] is the association list
|
||||
containing the bindings of the map [map], sorted by increasing keys
|
||||
(with respect to the total comparison function used to create the
|
||||
map). *)
|
||||
|
||||
val bindings : ('key, 'value) t -> ('key * 'value) list
|
||||
|
||||
(* The side-effect of evaluating the call [iter f map] is the
|
||||
successive side-effects of the calls [f key value], for all
|
||||
bindings [(key, value)] belonging to the map [map], sorted in
|
||||
increasing order of the keys (with respect to the total comparison
|
||||
function used to create the map). *)
|
||||
|
||||
val iter : ('key -> 'value -> unit) -> ('key, 'value) t -> unit
|
||||
|
||||
(* The call [fold_inc f map ~init] computes [(f k_n v_n ~acc:(... (f
|
||||
k_1 v_1 ~acc:init)...)], where [k_1], ..., [k_n] are the keys of
|
||||
all bindings in the map [map] in increasing order, and [v_1], ...,
|
||||
[v_n] are the associated values. *)
|
||||
|
||||
val fold_inc : ('key -> 'value -> acc:'a -> 'a) -> ('key, 'value) t -> init:'a -> 'a
|
59
vendors/Red-Black_Trees/PolyMapMain.ml
vendored
Normal file
59
vendors/Red-Black_Trees/PolyMapMain.ml
vendored
Normal file
@ -0,0 +1,59 @@
|
||||
(* Unit testing of module [PolyMap] *)
|
||||
|
||||
let () = Printf.printf "Testing polymorphic maps... "
|
||||
|
||||
let cmp = Pervasives.compare
|
||||
|
||||
let in_assoc =
|
||||
["c", 2;
|
||||
"g", -1;
|
||||
"a", 0;
|
||||
"e", 4;
|
||||
"f", 5;
|
||||
"g", -1;
|
||||
"b", -1;
|
||||
"b", 1;
|
||||
"d", 3;
|
||||
"j", 9;
|
||||
"g", 6;
|
||||
"h", 7;
|
||||
"i", 8]
|
||||
|
||||
(* Insertion sort which keeps the last duplicate of any item.
|
||||
WARNING: Quadratic cost! *)
|
||||
|
||||
let rec ins cmp x = function
|
||||
[] -> [x]
|
||||
| y::s as l ->
|
||||
let diff = cmp x y in
|
||||
if diff = 0 then l else if diff > 0 then y :: ins cmp x s else x::l
|
||||
|
||||
let rec isort cmp = function
|
||||
[] -> []
|
||||
| x::s -> ins cmp x (isort cmp s)
|
||||
|
||||
|
||||
let sorted_in_assoc = isort (fun (k1,_) (k2,_) -> cmp k1 k2) in_assoc
|
||||
|
||||
(*
|
||||
let () = Printf.printf "\nInput map:\n"
|
||||
let () = List.iter (fun (k,v) -> Printf.printf "%s -> %d\n" k v) in_assoc
|
||||
let () = Printf.printf "\nSorted map:\n"
|
||||
let () = List.iter (fun (k,v) -> Printf.printf "%s -> %d\n" k v) sorted_in_assoc
|
||||
*)
|
||||
|
||||
let empty_map = PolyMap.create ~cmp
|
||||
let out_map =
|
||||
List.fold_left (fun m (k,v) -> PolyMap.add k v m) empty_map in_assoc
|
||||
|
||||
let out_bindings = PolyMap.bindings out_map
|
||||
|
||||
(*
|
||||
let () = Printf.printf "\nOutput map:\n"
|
||||
let () = List.iter (fun (k,v) -> Printf.printf "%s -> %d\n" k v) out_bindings
|
||||
*)
|
||||
|
||||
let () =
|
||||
if sorted_in_assoc = out_bindings then
|
||||
Printf.printf "PASS.\n%!"
|
||||
else Printf.printf "FAILED.\n%!"
|
32
vendors/Red-Black_Trees/PolySet.ml
vendored
Normal file
32
vendors/Red-Black_Trees/PolySet.ml
vendored
Normal file
@ -0,0 +1,32 @@
|
||||
(* Polymorphic sets *)
|
||||
|
||||
module RB = RedBlack
|
||||
|
||||
type 'elt t = {
|
||||
tree : 'elt RB.t;
|
||||
cmp : 'elt -> 'elt -> int
|
||||
}
|
||||
|
||||
type 'elt set = 'elt t
|
||||
|
||||
let create ~cmp = {tree = RB.empty; cmp}
|
||||
|
||||
let empty = {tree = RB.empty; cmp=Pervasives.compare}
|
||||
|
||||
let is_empty set = RB.is_empty set.tree
|
||||
|
||||
let add elt set = {set with tree = RB.add ~cmp:set.cmp RB.New elt set.tree}
|
||||
|
||||
exception Not_found
|
||||
|
||||
let find elt set =
|
||||
try RB.find ~cmp:set.cmp elt set.tree with
|
||||
RB.Not_found -> raise Not_found
|
||||
|
||||
let find_opt elt set = RB.find_opt ~cmp:set.cmp elt set.tree
|
||||
|
||||
let elements set = RB.elements set.tree
|
||||
|
||||
let iter f set = RB.iter f set.tree
|
||||
|
||||
let fold_inc f set = RB.fold_inc (fun ~elt -> f elt) set.tree
|
70
vendors/Red-Black_Trees/PolySet.mli
vendored
Normal file
70
vendors/Red-Black_Trees/PolySet.mli
vendored
Normal file
@ -0,0 +1,70 @@
|
||||
(* Polymorphic ordered sets
|
||||
|
||||
This module does not provide a function to merge polymorphic
|
||||
sets. Use the functorial interface of the module [Set] of the OCaml
|
||||
standard library instead.
|
||||
|
||||
No deletion is provided.
|
||||
*)
|
||||
|
||||
type 'elt t
|
||||
type 'elt set = 'elt t
|
||||
|
||||
(* The value of the call [create ~cmp] is an empty set with [cmp]
|
||||
being the comparison over the (future) keys.
|
||||
|
||||
The value [empty] is identical to the value of the call [create
|
||||
~cmp:Pervasives.compare].
|
||||
*)
|
||||
|
||||
val create : cmp:('elt -> 'elt -> int) -> 'elt t
|
||||
|
||||
val empty : 'elt t
|
||||
|
||||
(* Emptiness *)
|
||||
|
||||
val is_empty : 'elt t -> bool
|
||||
|
||||
(* The value of the call [add elt set] is the union of the set [set]
|
||||
and the singleton set containing [elt]. If there is an element [y]
|
||||
in [set] such that [cmp y elt = true], where [cmp] is the
|
||||
comparison function of the set [set] (see [create]), then [y] is
|
||||
replaced by [elt]. *)
|
||||
|
||||
val add : 'elt -> 'elt t -> 'elt t
|
||||
|
||||
(* The value of the call [find elt set] is the element [y] of set
|
||||
[set] such that [cmp y elt = true], where [cmp] is the comparison
|
||||
function of [set] (see [create]). If [elt] is not in [set], then
|
||||
the exception [Not_found] is raised. *)
|
||||
|
||||
exception Not_found
|
||||
|
||||
val find : 'elt -> 'elt t -> 'elt
|
||||
|
||||
(* The call [find_opt elt set] is similar to [find elt set], except
|
||||
that [None] is returned instead of the [Not_found] exception being
|
||||
raised, otherwise it is an optional element. *)
|
||||
|
||||
val find_opt : 'elt -> 'elt t -> 'elt option
|
||||
|
||||
(* The value of the call [element set] is the list of elements of the
|
||||
set [set] in increasing order (with respect to the total comparison
|
||||
function used to create the set). *)
|
||||
|
||||
val elements : 'elt t -> 'elt list
|
||||
|
||||
(* The side-effect of evaluating the call [iter f set] is the
|
||||
successive side-effects of the calls [f elt], for all the elements
|
||||
[elt] of the set [set], sorted in increasing order (with respect to
|
||||
the total comparison function used to create the set). *)
|
||||
|
||||
val iter : ('elt -> unit) -> 'elt t -> unit
|
||||
|
||||
(* The value of the call [fold_inc f set ~init] is the result of
|
||||
iterating the function [f] on all the elements of the set [set] in
|
||||
increasing order (with respect to the total comparison function
|
||||
used to create the set), accumulating partial results from the
|
||||
initial value [init]. *)
|
||||
|
||||
val fold_inc : ('elt -> acc:'a -> 'a) -> 'elt t -> init:'a -> 'a
|
34
vendors/Red-Black_Trees/PolySetMain.ml
vendored
Normal file
34
vendors/Red-Black_Trees/PolySetMain.ml
vendored
Normal file
@ -0,0 +1,34 @@
|
||||
(* Unit testing of module [PolySet] *)
|
||||
|
||||
open PolySet;;
|
||||
|
||||
let () = Printf.printf "Testing polymorphic sets... "
|
||||
|
||||
let cmp = Pervasives.compare
|
||||
|
||||
let in_items = [6;7;9;1;0;3;6;1;8;5;4;2]
|
||||
let sorted_in_items = List.sort_uniq cmp in_items
|
||||
|
||||
(*
|
||||
let () = Printf.printf "\nInput items: "
|
||||
let () = List.iter (fun x -> Printf.printf "%d " x) in_items
|
||||
let () = Printf.printf "\nSorted input: "
|
||||
let () = List.iter (fun x -> Printf.printf "%d " x) sorted_in_items
|
||||
*)
|
||||
|
||||
let empty_set = create ~cmp
|
||||
|
||||
let set = List.fold_right add in_items empty_set
|
||||
let out_items = elements set
|
||||
|
||||
(*
|
||||
let () = Printf.printf "\nOutput items: "
|
||||
let () = List.iter (fun x -> Printf.printf "%d " x) out_items
|
||||
let () = Printf.printf "\n%!"
|
||||
*)
|
||||
|
||||
let () =
|
||||
if sorted_in_items = out_items then
|
||||
Printf.printf "PASS.\n%!"
|
||||
else Printf.printf "FAILED.\n%!"
|
||||
|
5
vendors/Red-Black_Trees/README.md
vendored
Normal file
5
vendors/Red-Black_Trees/README.md
vendored
Normal file
@ -0,0 +1,5 @@
|
||||
# Read-Black trees a la Okasaki and polymorphic sets and maps based on them
|
||||
|
||||
This implementation is based on the classic paper by Chris Okasaki,
|
||||
*Red-Black Trees in a Functional Setting*. J. Funct. Program. 9(4):
|
||||
471-477 (1999).
|
86
vendors/Red-Black_Trees/RedBlack.ml
vendored
Normal file
86
vendors/Red-Black_Trees/RedBlack.ml
vendored
Normal file
@ -0,0 +1,86 @@
|
||||
(* Red-black trees according to the following classic paper:
|
||||
|
||||
Chris Okasaki, Red-Black Trees in a Functional
|
||||
Setting. J. Funct. Program. 9(4): 471-477 (1999)
|
||||
*)
|
||||
|
||||
type colour = Red | Black
|
||||
|
||||
type 'a t =
|
||||
Ext
|
||||
| Int of colour * 'a t * 'a * 'a t
|
||||
|
||||
let empty = Ext
|
||||
|
||||
let is_empty m = (m = empty)
|
||||
|
||||
let blacken = function
|
||||
Ext -> Ext
|
||||
| Int (_, left, root, right) -> Int (Black, left, root, right)
|
||||
|
||||
let balance colour left root right =
|
||||
match colour, left, root, right with
|
||||
Black, Int (Red, Int (Red, a, x, b), y, c), z, d
|
||||
| Black, Int (Red, a, x, Int (Red, b, y, c)), z, d
|
||||
| Black, a, x, Int (Red, Int (Red, b, y, c), z, d)
|
||||
| Black, a, x, Int (Red, b, y, Int (Red, c, z, d)) ->
|
||||
Int (Red, Int (Black, a, x, b), y, Int (Black, c, z, d))
|
||||
| _ -> Int (colour, left, root, right)
|
||||
|
||||
type choice = Old | New
|
||||
|
||||
let choose ~old ~new' = function
|
||||
Old -> old
|
||||
| New -> new'
|
||||
|
||||
exception Physical_equality
|
||||
|
||||
let add ~cmp choice elt tree =
|
||||
let rec insert = function
|
||||
Ext -> Int (Red, Ext, elt, Ext) (* A leaf *)
|
||||
| Int (colour, left, root, right) ->
|
||||
let diff = cmp elt root in
|
||||
if diff = 0 then
|
||||
let root' = choose ~new':elt ~old:root choice
|
||||
in if root == root' then raise Physical_equality
|
||||
else Int (colour, left, root', right)
|
||||
else if diff < 0 then
|
||||
balance colour (insert left) root right
|
||||
else balance colour left root (insert right)
|
||||
in try blacken (insert tree) with
|
||||
Physical_equality -> tree
|
||||
|
||||
exception Not_found
|
||||
|
||||
let rec find ~cmp elt = function
|
||||
Ext -> raise Not_found
|
||||
| Int (_, left, root, right) ->
|
||||
let diff = cmp elt root in
|
||||
if diff = 0 then root
|
||||
else if diff < 0 then find ~cmp elt left
|
||||
else find ~cmp elt right
|
||||
|
||||
let find_opt ~cmp elt tree =
|
||||
try Some (find ~cmp elt tree) with Not_found -> None
|
||||
|
||||
(* Inorder iterators *)
|
||||
|
||||
let rec iter f = function
|
||||
Ext -> ()
|
||||
| Int (_, left, root, right) -> iter f left; f root; iter f right
|
||||
|
||||
let rec inorder acc = function
|
||||
Ext -> acc
|
||||
| Int (_, left, root, right) -> inorder (root :: inorder acc right) left
|
||||
|
||||
let elements t = inorder [] t
|
||||
|
||||
let rec fold_inc f ~init = function
|
||||
Ext -> init
|
||||
| Int (_, left, root, right) ->
|
||||
fold_inc f ~init:(f ~elt:root ~acc:(fold_inc f ~init left)) right
|
||||
|
||||
let rec fold_dec f ~init = function
|
||||
Ext -> init
|
||||
| Int (_, left, root, right) ->
|
||||
fold_dec f ~init:(f ~elt:root ~acc:(fold_dec f ~init right)) left
|
66
vendors/Red-Black_Trees/RedBlack.mli
vendored
Normal file
66
vendors/Red-Black_Trees/RedBlack.mli
vendored
Normal file
@ -0,0 +1,66 @@
|
||||
(* Red-black trees according to the following classic paper:
|
||||
|
||||
Chris Okasaki, Red-Black Trees in a Functional
|
||||
Setting. J. Funct. Program. 9(4): 471-477 (1999)
|
||||
*)
|
||||
|
||||
type colour = Red | Black
|
||||
|
||||
type 'a t = private
|
||||
Ext
|
||||
| Int of colour * 'a t * 'a * 'a t
|
||||
|
||||
val empty: 'a t
|
||||
|
||||
val is_empty: 'a t -> bool
|
||||
|
||||
(* The value of the call [add ~cmp choice x t] is a red-black tree
|
||||
augmenting the tree [t] with a node containing the element [x],
|
||||
using the comparison function [cmp] (following the same convention
|
||||
as [Pervasives.compare]) and, if a value [y] such that [x = y] is
|
||||
already present in a node of [t], then the value [choice] denotes
|
||||
whether [x] ([New]) or [y] ([Old]) remains in the value of the
|
||||
call. Moreover, if [x == y], then [add ~cmp choice x t == t]. *)
|
||||
|
||||
type choice = Old | New
|
||||
|
||||
val add: cmp:('a -> 'a -> int) -> choice -> 'a -> 'a t -> 'a t
|
||||
|
||||
(* The value of the call [find ~cmp x t] is the element [y] belonging
|
||||
to a node of the tree [t], such that [cmp x y = true]. If none, the
|
||||
exception [Not_found] is raised. *)
|
||||
|
||||
exception Not_found
|
||||
|
||||
val find: cmp:('a -> 'b -> int) -> 'a -> 'b t -> 'b
|
||||
|
||||
(* The value of call [find_opt ~cmp x t] is [Some y] if there is an
|
||||
element [y] in a node of the tree [t] such that [cmp x y = true],
|
||||
and [None] otherwise. *)
|
||||
|
||||
val find_opt: cmp:('a -> 'b -> int) -> 'a -> 'b t -> 'b option
|
||||
|
||||
(* The value of the call [elements t] is the list of elements in the
|
||||
nodes of the tree [t], sorted by increasing order. *)
|
||||
|
||||
val elements: 'a t -> 'a list
|
||||
|
||||
(* The side-effect of evaluating the call [iter f t] is the successive
|
||||
side-effects of the calls [f x], for all elements [x] belonging to
|
||||
the nodes of [t], visited in increasing order. *)
|
||||
|
||||
val iter: ('a -> unit) -> 'a t -> unit
|
||||
|
||||
(* The value of the call [fold_inc f ~init t] is the iteration of the
|
||||
function [f] on increasing elements of the nodes of tree [t],
|
||||
accumulating the partial results from the initial value of
|
||||
[init]. *)
|
||||
|
||||
val fold_inc: (elt:'a -> acc:'b -> 'b) -> init:'b -> 'a t -> 'b
|
||||
|
||||
(* The value of the call [fold_dec f ~init t] is the iteration of the
|
||||
function [f] on decreasing elements of the nodes of tree [t],
|
||||
accumulating the partial results from the initial value of
|
||||
[init]. *)
|
||||
|
||||
val fold_dec: (elt:'a -> acc:'b -> 'b) -> init:'b -> 'a t -> 'b
|
88
vendors/Red-Black_Trees/RedBlackMain.ml
vendored
Normal file
88
vendors/Red-Black_Trees/RedBlackMain.ml
vendored
Normal file
@ -0,0 +1,88 @@
|
||||
(* Unit testing of module [RedBlack] *)
|
||||
|
||||
open RedBlack;;
|
||||
|
||||
let () = Printf.printf "Testing Red-black trees... "
|
||||
|
||||
let cmp = Pervasives.compare
|
||||
|
||||
let in_items = [6;7;9;1;0;3;6;1;8;5;4;2]
|
||||
let sorted_in_items = List.sort_uniq cmp in_items
|
||||
|
||||
let () = Printf.printf "\nInput items: "
|
||||
let () = List.iter (fun x -> Printf.printf "%d " x) in_items
|
||||
let () = Printf.printf "\nSorted input: "
|
||||
let () = List.iter (fun x -> Printf.printf "%d " x) sorted_in_items
|
||||
|
||||
let t = List.fold_right (add ~cmp Old) in_items empty
|
||||
let out_items = elements t
|
||||
|
||||
let () = Printf.printf "\nOutput items: "
|
||||
let () = List.iter (fun x -> Printf.printf "%d " x) out_items
|
||||
let () = Printf.printf "\n%!"
|
||||
|
||||
let () =
|
||||
if sorted_in_items = out_items then
|
||||
Printf.printf "PASS.\n%!"
|
||||
else Printf.printf "FAILED.\n%!"
|
||||
|
||||
let rec draw ~pad:(pd,pc) = function
|
||||
Ext -> ()
|
||||
| Int (colour, left, root, right) ->
|
||||
let app i sub =
|
||||
let pad =
|
||||
(pc ^ (if i = 1 then "`-- " else "|-- "),
|
||||
pc ^ (if i = 1 then " " else "| "))
|
||||
in draw ~pad sub in
|
||||
begin
|
||||
Printf.printf "%s%s(%d)\n" pd (if colour = Red then "R" else "B") root;
|
||||
List.iteri app [left; right]
|
||||
end
|
||||
|
||||
let () = draw ~pad:("","") t
|
||||
|
||||
let rec to_string buffer ~pad:(pd,pc) = function
|
||||
Ext -> ()
|
||||
| Int (colour, left, root, right) ->
|
||||
let root_str =
|
||||
Printf.sprintf "%s%s(%d)\n" pd
|
||||
(if colour = Red then "R" else "B") root in
|
||||
let app rank sub =
|
||||
let pad =
|
||||
(pc ^ (if rank = 1 then "`-- " else "|-- "),
|
||||
pc ^ (if rank = 1 then " " else "| "))
|
||||
in to_string buffer ~pad sub in
|
||||
begin
|
||||
Buffer.add_string buffer root_str;
|
||||
List.iteri app [left; right]
|
||||
end
|
||||
|
||||
let to_string tree =
|
||||
let buffer = Buffer.create 131 in
|
||||
let () = to_string buffer ~pad:("","") tree
|
||||
in Buffer.contents buffer
|
||||
|
||||
let () = to_string t |> print_string |> print_newline
|
||||
|
||||
let rec pretty buffer ~pad:(pd,pc) = function
|
||||
Ext -> Buffer.add_string buffer (pd ^ "Ext\n")
|
||||
| Int (colour, left, root, right) ->
|
||||
let root_str =
|
||||
Printf.sprintf "%sInt (%s,%d)\n" pd
|
||||
(if colour = Red then "Red" else "Black") root in
|
||||
let app rank sub =
|
||||
let pad =
|
||||
pc ^ (if rank = 0 then "|-- " else "`-- "),
|
||||
pc ^ (if rank = 0 then "| " else " ")
|
||||
in pretty buffer ~pad sub in
|
||||
begin
|
||||
Buffer.add_string buffer root_str;
|
||||
List.iteri app [left; right]
|
||||
end
|
||||
|
||||
let pretty tree =
|
||||
let buffer = Buffer.create 131 in
|
||||
let () = pretty buffer ~pad:("","") tree
|
||||
in Buffer.contents buffer
|
||||
|
||||
let () = pretty t |> print_string |> print_newline
|
1
vendors/Red-Black_Trees/URL
vendored
Normal file
1
vendors/Red-Black_Trees/URL
vendored
Normal file
@ -0,0 +1 @@
|
||||
https://github.com/rinderknecht/Red-Black_Trees/commit/98b11312810d979ee1b0dacee6900228574e41e7
|
17
vendors/Red-Black_Trees/build.sh
vendored
Executable file
17
vendors/Red-Black_Trees/build.sh
vendored
Executable file
@ -0,0 +1,17 @@
|
||||
#!/bin/sh
|
||||
set -x
|
||||
ocamlfind ocamlc -strict-sequence -w +A-48-4 -c RedBlack.mli
|
||||
ocamlfind ocamlc -strict-sequence -w +A-48-4 -c PolyMap.mli
|
||||
ocamlfind ocamlopt -strict-sequence -w +A-48-4 -c RedBlack.ml
|
||||
ocamlfind ocamlopt -strict-sequence -w +A-48-4 -c PolyMapMain.ml
|
||||
ocamlfind ocamlopt -strict-sequence -w +A-48-4 -c PolyMap.ml
|
||||
ocamlfind ocamlopt -strict-sequence -w +A-48-4 -c PolyMapMain.ml
|
||||
ocamlfind ocamlopt -o PolyMapMain.opt RedBlack.cmx PolyMap.cmx PolyMapMain.cmx
|
||||
ocamlfind ocamlc -strict-sequence -w +A-48-4 -c PolySet.mli
|
||||
ocamlfind ocamlopt -strict-sequence -w +A-48-4 -c PolySet.ml
|
||||
ocamlfind ocamlopt -strict-sequence -w +A-48-4 -c PolySetMain.ml
|
||||
ocamlfind ocamlopt -strict-sequence -w +A-48-4 -c PolySetMain.ml
|
||||
ocamlfind ocamlopt -o PolySetMain.opt RedBlack.cmx PolySet.cmx PolySetMain.cmx
|
||||
ocamlfind ocamlopt -strict-sequence -w +A-48-4 -c RedBlackMain.ml
|
||||
ocamlfind ocamlopt -strict-sequence -w +A-48-4 -c RedBlackMain.ml
|
||||
ocamlfind ocamlopt -o RedBlackMain.opt RedBlack.cmx RedBlackMain.cmx
|
3
vendors/Red-Black_Trees/clean.sh
vendored
Executable file
3
vendors/Red-Black_Trees/clean.sh
vendored
Executable file
@ -0,0 +1,3 @@
|
||||
#!/bin/sh
|
||||
|
||||
\rm -f *.cm* *.o *.byte *.opt
|
Loading…
Reference in New Issue
Block a user