File: prolog.ml

package info (click to toggle)
facile 1.1.4-4
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid, trixie
  • size: 716 kB
  • sloc: ml: 6,862; makefile: 90
file content (129 lines) | stat: -rw-r--r-- 4,115 bytes parent folder | download | duplicates (9)
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
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
(***********************************************************************)
(*                                                                     *)
(*                           FaCiLe                                    *)
(*                 A Functional Constraint Library                     *)
(*                                                                     *)
(*            Nicolas Barnier, Pascal Brisset, LOG, CENA               *)
(*                                                                     *)
(* Copyright 2004 CENA. All rights reserved. This file is distributed  *)
(* under the terms of the GNU Lesser General Public License.           *)
(***********************************************************************)
(* $Id: prolog.ml,v 1.8 2001/06/15 10:27:27 barnier Exp $ *)

(* FaCiLe as a Prolog interpreter: the family tree

   In this example, we write the classic goals parent, grandparent and ancestor
   for the following family tree:

   sam
    |
   jim + lucy
       |
     fred + lynn
          |
         ann

   Then we search all the solutions for various questions, using a "findall"
   goal that builds a list of all the possible values for a given variable such
   that a given goal succeeds.

   This example was inspired by Mattias Waldau and translated from the
   following Prolog source:

   father(ann, fred).
   father(fred, jim).
   father(jim, sam).

   mother(ann, lynn).
   mother(fred, lucy).

   parent(X,Y) :- mother(X,Y).
   parent(X,Y) :- father(X,Y).

   grandparent(X,Y) :- parent(X,Z), parent(Z,Y).

   ancestor(X,Y) :- parent(X,Y).
   ancestor(X,Y) :- parent(X,Z), ancestor(Z,Y).
*)

open Facile
open Easy

let ann = 0 and fred = 1 and jim = 2
and sam = 3 and lynn = 4 and lucy = 5

let name_of = function
    0 -> "Ann" | 1 -> "Fred" | 2 -> "Jim"
  | 3 -> "Sam" | 4 -> "Lynn" | 5 -> "Lucy" | _ -> invalid_arg "name_of"

let family_dom = Domain.create [ann; fred; jim; sam; lynn; lucy]

(* child first, parent second *)
let fathers = [(ann, fred); (fred, jim); (jim, sam)]
let mothers = [(ann, lynn); (fred, lucy)]

(* [father x y] [x]: child; [y]: father*)
let father x y =
  Goals.List.exists (fun (c, f) -> Goals.unify x c &&~ Goals.unify y f) fathers

(* [mother x y] [x]: child; [y]: mother] *)
let mother x y =
  Goals.List.exists (fun (c, m) -> Goals.unify x c &&~ Goals.unify y m) mothers

let parent x y = father x y ||~ mother x y

(* We use the existential quantifier [Goals.sigma] to hide the creation of
   an intermediate variable. However, we provide its domain [family_dom] which
   is an optional argument (if omitted, the domain is the largest one). *)
let grandparent x y =
  Goals.sigma ~domain:family_dom (fun z -> parent x z &&~ parent z y)

(* Recursive goal implemented from a recursive function using Goals.create *)
let rec ancestor x y =
  Goals.create
    (fun () ->
      parent x y ||~
      (Goals.sigma ~domain:family_dom (fun z -> parent x z &&~ ancestor z y)))
    ()

(* [val findall : (Fd.t -> Goals.t) -> int list]
   [findall g] returns all the solutions of variables [y] such that the goal [g y]
   succeeds. *)
let findall g =
  (* The solutions are stored  in a list. *)
  let sol = ref [] in
  let store v = Goals.atomic (fun () -> sol := Fd.int_value v :: !sol) in
  let goal =
    Goals.sigma ~domain:family_dom (fun y -> g y &&~ store y &&~ Goals.fail)
     ||~
    Goals.success in
  if Goals.solve goal then
    !sol
  else
    failwith "Unexpected failure"

let all_ancestors x =
  findall (fun y -> ancestor x y)

let all_parents x =
  findall (fun y -> parent x y)

let all_grandchildren y =
  findall (fun x -> grandparent x y)

let _ =
  let print_list l =
    List.iter (fun a -> Printf.printf "%s " (name_of a)) l;
    print_newline () in
  (* All the ancestors of Ann *)
  let ancestors = all_ancestors (Fd.int ann) in
  print_list ancestors;

  (* All the parents of Jim and Ann *)
  let parents = all_parents (Fd.create (Domain.create [jim; ann])) in
  print_list parents;

  (* All the grandchildren of Sam *)
  let grandchildren = all_grandchildren (Fd.int sam) in
  print_list grandchildren