Page
Library
Module
Module type
Parameter
Class
Class type
Source
ChoiceThis is an attempt to implement the Logic monad, as described in this Haskell library or in this paper.
Some design choices are slightly different, since OCaml is strict. Performance may also vary from ghc-compiled code, since it performs some optimizations like deforestation.
We adapt the example from the LogicT paper.
let rec insert e l = match l with
| [] -> return [e]
| x::l' ->
mplus
(return (e :: l))
(insert e l' >>= fun l'' -> return (x :: l''));;
let rec permute = function
| [] -> return []
| x::l ->
permute l >>= fun l' ->
insert x l';;
let rec sorted l = match l with
| [] | [_] -> true
| x::((y::l') as l) ->
x <= y && sorted l;;
let bogosort l =
once (filter sorted (permute l));;Then, running
run_n 1 (bogosort [2;3;5;1;0]);; yields
- : int list list = [[0; 1; 2; 3; 5]] type 'a choice = 'a tval return : 'a -> 'a tReturn a value, and succeed
val of_list : 'a list -> 'a tMultiple returns. Each element of the list is a candidate for success.
val from_fun : (unit -> 'a option) -> 'a tCall the function to get alternative choices. Example:
let r = ref 0 in Choice.run_n 10
(Choice.filter
(Choice.from_fun (fun () -> incr r; Some !r)) (fun x -> x mod 3 = 0));;yields
- : int list = [30; 27; 24; 21; 18; 15; 12; 9; 6; 3] Delay the computation (the closure will be called in each branch that uses the choice point
val fail : 'a tFail to yield a solution.
Monadic bind. Each solution of the first argument is given to the function, that may in turn return several choices.
Same as mplus, but fair, ie it enumerates solutions alternatively from its first and second arguments.
ite cond th el enumerates the choices of cond. If cond fails, then it behaves like el, otherwise each solution of cond is given to th.
Special case of bind, with only zero or one possible output choices for each input choice.
val run_one : 'a t -> 'a optionRun until we get one answer (or a failure)
val run_n : int -> 'a t -> 'a listThe n first solutions, in reverse order.
val run_all : 'a t -> 'a listAll the solutions (in reverse order)
val to_list : 'a t -> 'a listAll the solutions (in correct order)
val iter : 'a t -> ('a -> bool) -> unitEnumerate solutions, until none remains, or the callback returns false to signal it has had enough solutions
val fold : ('a -> 'b -> 'a) -> 'a -> 'b t -> 'aFold over all solutions
val count : _ t -> intNumber of solutions
val is_empty : _ t -> boolreturn true iff the alternative stream is empty (failure)
val forall : bool t -> boolval exists : bool t -> boolval pure : ('a -> 'b) -> ('a -> 'b) tInfix version of interleave
module Enum : sig ... endmodule List : sig ... endmodule Array : sig ... end