csci2041/public-class-repo/SamplePrograms/Intervals/v7/intervals.ml

82 lines
2.3 KiB
OCaml
Raw Normal View History

2018-01-29 23:35:31 +00:00
(* As before, we need to signature for endpoints. *)
module type Comparable = sig
type t
val compare : t -> t -> int
val to_string : t -> string
end
(* Now, define the signature that hide the type of the interval, named
t below, and the type of the end points, named endpoint.
*)
module type Interval_intf = sig
type t
type endpoint
val create : endpoint -> endpoint -> t
val is_empty : t -> bool
val contains : t -> endpoint -> bool
val intersect : t -> t -> t
val to_string : t -> string
end
(* The Make_interval functor takes a module that matches the
Comparable signature and uses it to create an interval module over
endpoint defined by that input module.
****
It also states the the 'endpoint' type in 'Interval_intf'
is replaced by hee type 't' in the 'Endpoint' module passed
in to 'Make_interval'
In this case we used := instead of = to indicate this destructive
substitution.
*****
*)
module Make_interval(Endpoint : Comparable) :
(Interval_intf with type endpoint := Endpoint.t) = struct
type endpoint = Endpoint.t
type t = | Interval of Endpoint.t * Endpoint.t
| Empty
(** [create low high] creates a new interval from [low] to
[high]. If [low > high], then the interval is empty *)
let create low high =
if Endpoint.compare low high > 0 then Empty
else Interval (low,high)
(** Returns true iff the interval is empty *)
let is_empty = function
| Empty -> true
| Interval _ -> false
(** [contains t x] returns true iff [x] is contained in the
interval [t] *)
let contains t x =
match t with
| Empty -> false
| Interval (l,h) ->
Endpoint.compare x l >= 0 && Endpoint.compare x h <= 0
(** [intersect t1 t2] returns the intersection of the two input
intervals *)
let intersect t1 t2 =
let min x y = if Endpoint.compare x y <= 0 then x else y in
let max x y = if Endpoint.compare x y >= 0 then x else y in
match t1,t2 with
| Empty, _ | _, Empty -> Empty
| Interval (l1,h1), Interval (l2,h2) ->
create (max l1 l2) (min h1 h2)
let to_string i : string =
match i with
| Empty -> "Empty"
| Interval (l,h)
-> "(" ^ Endpoint.to_string l ^ ", " ^ Endpoint.to_string h ^ ")"
end