239 lines
6.4 KiB
OCaml
239 lines
6.4 KiB
OCaml
|
type value
|
||
|
= Int of int
|
||
|
| Bool of bool
|
||
|
|
||
|
type expr =
|
||
|
| Add of expr * expr
|
||
|
| Mul of expr * expr
|
||
|
| Sub of expr * expr
|
||
|
| Div of expr * expr
|
||
|
| Mod of expr * expr
|
||
|
|
||
|
| Lt of expr * expr
|
||
|
| Eq of expr * expr
|
||
|
| And of expr * expr
|
||
|
|
||
|
| Var of string
|
||
|
| Value of value
|
||
|
|
||
|
type environment = (string * value) list
|
||
|
|
||
|
let rec lookup name env =
|
||
|
match env with
|
||
|
| [ ] -> raise (Failure ("Name \"" ^ name ^ "\" not found."))
|
||
|
| (k,v)::rest -> if name = k then v else lookup name rest
|
||
|
|
||
|
let rec eval (e: expr) (env: environment) : value =
|
||
|
match e with
|
||
|
| Value v -> v
|
||
|
| Add (e1, e2) ->
|
||
|
( match eval e1 env, eval e2 env with
|
||
|
| Int v1, Int v2 -> Int (v1 + v2)
|
||
|
| _ -> raise (Failure "incompatible types, Add")
|
||
|
)
|
||
|
| Sub (e1, e2) ->
|
||
|
( match eval e1 env, eval e2 env with
|
||
|
| Int v1, Int v2 -> Int (v1 - v2)
|
||
|
| _ -> raise (Failure "incompatible types, Sub")
|
||
|
)
|
||
|
| Mul (e1, e2) ->
|
||
|
( match eval e1 env, eval e2 env with
|
||
|
| Int v1, Int v2 -> Int (v1 * v2)
|
||
|
| _ -> raise (Failure "incompatible types, Mul")
|
||
|
)
|
||
|
| Div (e1, e2) ->
|
||
|
( match eval e1 env, eval e2 env with
|
||
|
| Int v1, Int v2 -> Int (v1 / v2)
|
||
|
| _ -> raise (Failure "incompatible types, Div")
|
||
|
)
|
||
|
| Mod (e1, e2) ->
|
||
|
(match eval e1 env, eval e2 env with
|
||
|
| Int v1, Int v2 -> Int (v1 mod v2)
|
||
|
| _ -> raise (Failure "incompatible types, Mod")
|
||
|
)
|
||
|
| Lt (e1, e2) ->
|
||
|
( match eval e1 env, eval e2 env with
|
||
|
| Int v1, Int v2 -> Bool (v1 < v2)
|
||
|
| _ -> raise (Failure "incompatible types, Lt")
|
||
|
)
|
||
|
| Eq (e1, e2) ->
|
||
|
( match eval e1 env, eval e2 env with
|
||
|
| Int v1, Int v2 -> Bool (v1 = v2)
|
||
|
| Bool v1, Bool v2 -> Bool (v1 = v2)
|
||
|
| _ -> raise (Failure "incompatible types, Eq")
|
||
|
)
|
||
|
| And (e1, e2) ->
|
||
|
( match eval e1 env, eval e2 env with
|
||
|
| Bool v1, Bool v2 -> Bool (v1 && v2)
|
||
|
| _ -> raise (Failure "incompatible types, And")
|
||
|
)
|
||
|
| Var n -> lookup n env
|
||
|
|
||
|
|
||
|
|
||
|
type state = environment
|
||
|
|
||
|
type stmt =
|
||
|
| Assign of string * expr
|
||
|
| While of expr * stmt
|
||
|
| IfThen of expr * stmt
|
||
|
| IfThenElse of expr * stmt * stmt
|
||
|
| Seq of stmt * stmt
|
||
|
| WriteNum of expr
|
||
|
| ReadNum of string
|
||
|
| Skip
|
||
|
|
||
|
(* x := 1;
|
||
|
y := x + 2;
|
||
|
z := y + 3;
|
||
|
write z
|
||
|
*)
|
||
|
let program_1 =
|
||
|
Seq ( Assign ("x", Value (Int 1)) ,
|
||
|
Seq ( Assign ("y", Add (Var "x", Value (Int 2))),
|
||
|
Seq ( Assign ("z", Add (Var "y", Value (Int 3))),
|
||
|
WriteNum (Var "z")
|
||
|
)
|
||
|
)
|
||
|
)
|
||
|
|
||
|
|
||
|
(* read x;
|
||
|
i = 0;
|
||
|
sum = 0;
|
||
|
while (i < x) {
|
||
|
write i;
|
||
|
sum = sum + i;
|
||
|
i = i + 1
|
||
|
}
|
||
|
write sum
|
||
|
*)
|
||
|
|
||
|
let program_2 =
|
||
|
Seq (ReadNum "x",
|
||
|
Seq (Assign ("i", Value (Int 0)),
|
||
|
Seq (Assign ("sum", Value (Int 0)),
|
||
|
Seq (While (Lt (Var "i", Var "x"),
|
||
|
Seq (WriteNum (Var "i"),
|
||
|
Seq (Assign ("sum", Add (Var "sum", Var "i")),
|
||
|
Assign ("i", Add (Var "i", Value (Int 1)))
|
||
|
) ) ),
|
||
|
WriteNum (Var "sum")
|
||
|
) ) ) )
|
||
|
|
||
|
|
||
|
(* program_3
|
||
|
|
||
|
read x;
|
||
|
i = 0;
|
||
|
sum_evens = 0;
|
||
|
sum_odds = 0;
|
||
|
while (i < x) {
|
||
|
write i;
|
||
|
if i mod 2 = 0 then
|
||
|
sum_evens = sum_evens + i;
|
||
|
else
|
||
|
sum_odds = sum_odds + i;
|
||
|
i = i + 1
|
||
|
}
|
||
|
write sum_evens;
|
||
|
write sum_odds
|
||
|
*)
|
||
|
|
||
|
let rec read_number () =
|
||
|
print_endline "Enter an integer value:" ;
|
||
|
try int_of_string (read_line ()) with
|
||
|
| Failure _ -> read_number ()
|
||
|
|
||
|
let write_number n = print_endline (string_of_int n)
|
||
|
|
||
|
|
||
|
let rec exec (s: stmt) (stt: state) : state =
|
||
|
match s with
|
||
|
| Assign (v, e) -> (v, (eval e stt)) :: stt
|
||
|
|
||
|
| Seq (s1, s2) -> exec s2 (exec s1 stt)
|
||
|
|
||
|
| WriteNum e -> (match eval e stt with
|
||
|
| Int v -> write_number v; stt
|
||
|
| _ -> raise (Failure "Only numeric values can be printed.")
|
||
|
)
|
||
|
|
||
|
| ReadNum v -> (v, Int (read_number ())) :: stt
|
||
|
|
||
|
| While (cond, body) ->
|
||
|
(match eval cond stt with
|
||
|
| Bool true -> exec (Seq (body, While (cond, body))) stt
|
||
|
(* the version in the Sec_10 directory is slighlty
|
||
|
different, but achieves the same results. *)
|
||
|
| Bool false -> stt
|
||
|
)
|
||
|
|
||
|
| IfThenElse (cond, body1, body2) ->
|
||
|
(match eval cond stt with
|
||
|
| Bool true -> exec body1 stt
|
||
|
| Bool false -> exec body2 stt
|
||
|
)
|
||
|
|
||
|
| Skip -> stt
|
||
|
| IfThen (cond, body) ->
|
||
|
(match eval cond stt with
|
||
|
| Bool true -> exec body stt
|
||
|
| Bool false -> exec Skip stt
|
||
|
)
|
||
|
|
||
|
|
||
|
let num_sums = let output = [("i", Int 10); ("sum", Int 45); ("i", Int 9); ("sum", Int 36); ("i", Int 8);
|
||
|
("sum", Int 28); ("i", Int 7); ("sum", Int 21); ("i", Int 6); ("sum", Int 15);
|
||
|
("i", Int 5); ("sum", Int 10); ("i", Int 4); ("sum", Int 6); ("i", Int 3);
|
||
|
("sum", Int 3); ("i", Int 2); ("sum", Int 1); ("i", Int 1); ("sum", Int 0);
|
||
|
("sum", Int 0); ("i", Int 0); ("x", Int 10)] in
|
||
|
List.fold_right (fun x y -> match x with (z, w) -> if z = "sum" then (y+1) else y) output 0
|
||
|
|
||
|
|
||
|
let program_3 =
|
||
|
Seq (ReadNum "x",
|
||
|
Seq (Assign ("i", Value (Int 0)),
|
||
|
Seq (Assign ("sum_evens", Value (Int 0)),
|
||
|
Seq (Assign ("sum_odds", Value (Int 0)),
|
||
|
Seq (While (Lt (Var "i", Var "x"),
|
||
|
Seq (WriteNum (Var "i"),
|
||
|
Seq (IfThenElse (Eq (Mod (Var "i", Value (Int 2)), Value (Int 0)),
|
||
|
Assign("sum_evens", Add (Var "sum_evens", Var "i")),
|
||
|
Assign("sum_odds", Add (Var "sum_odds", Var "i"))
|
||
|
),
|
||
|
Assign("i", Add (Var "i", Value (Int 1)))
|
||
|
))),
|
||
|
Seq (WriteNum (Var "sum_evens"),
|
||
|
WriteNum (Var "sum_odds")
|
||
|
))))))
|
||
|
|
||
|
let val_sum_evens = 56
|
||
|
let val_sum_odds = 49
|
||
|
let num_sum_evens = 9
|
||
|
let num_sum_odds = 8
|
||
|
|
||
|
let program_3_test =
|
||
|
Seq (Assign ("x", Value (Int 12)),
|
||
|
Seq (Assign ("i", Value (Int 0)),
|
||
|
Seq (Assign ("sum_evens", Value (Int 0)),
|
||
|
Seq (Assign ("sum_odds", Value (Int 0)),
|
||
|
Seq (While (Lt (Var "i", Var "x"),
|
||
|
Seq (WriteNum (Var "i"),
|
||
|
Seq (IfThenElse (Eq (Mod (Var "i", Value (Int 2)), Value (Int 0)),
|
||
|
Assign("sum_evens", Add (Var "sum_evens", Var "i")),
|
||
|
Assign("sum_odds", Add (Var "sum_odds", Var "i"))
|
||
|
),
|
||
|
Assign("i", Add (Var "i", Value (Int 1)))
|
||
|
))),
|
||
|
Seq (WriteNum (Var "sum_evens"),
|
||
|
WriteNum (Var "sum_odds")
|
||
|
))))))
|
||
|
|
||
|
let program_4 =
|
||
|
Seq (Assign ("y", Value (Int 0)),
|
||
|
Seq (IfThen (Eq (Mod (Var "x", Value (Int 2)), Value (Int 0)), Assign ("y", Add (Var "y", Value (Int 2)))),
|
||
|
Seq (IfThen (Eq (Mod (Var "x", Value (Int 3)), Value (Int 0)), Assign ("y", Add (Var "y", Value (Int 3)))),
|
||
|
Seq (IfThen (Eq (Mod (Var "x", Value (Int 4)), Value (Int 0)), Assign ("y", Add (Var "y", Value (Int 4)))),
|
||
|
Skip
|
||
|
))))
|