uann/ocaml_yacc/calc.ml

51 lines
1.3 KiB
OCaml
Raw Normal View History

2024-03-25 00:34:14 +08:00
exception Exc of string
2024-03-25 00:32:19 +08:00
(* File calc.ml *)
let rec ast_to_string ast = match ast with
| Ast.Leaf s -> s
2024-03-25 00:34:14 +08:00
| Ast.Int i -> string_of_int i
2024-03-25 00:32:19 +08:00
| Ast.Node ls -> "[" ^ String.concat " " (List.map ast_to_string ls) ^ "]"
;;
2024-03-25 00:34:14 +08:00
(** interpreter for interp_int*)
class interp_int x_init =
object (self)
val mutable x = x_init
method interp code = match code with
| Ast.Int x -> x
| Ast.Node [Ast.Leaf "+"; lhs; rhs] ->
let x1 = self#interp lhs in
let x2 = self#interp rhs in
x1+x2
| Ast.Node [Ast.Leaf "-"; lhs; rhs] ->
let x1 = self#interp lhs in
let x2 = self#interp rhs in
x1-x2
| Ast.Node [Ast.Leaf "-"; x] ->
let x1 = self#interp x in
0-x1
| Ast.Node [Ast.Leaf "%apply" ; Ast.Leaf "read"; Ast.Int 0] ->
let input = read_int_opt () in
match input with
| Some(y) -> y
| _ -> raise (Exc "input invalid number")
| _ -> raise (Exc "unsupported")
end;;
(**main body*)
2024-03-25 00:32:19 +08:00
let _ =
try
let lexbuf = Lexing.from_channel stdin in
2024-03-25 00:34:14 +08:00
let interp = new interp_int 0 in
2024-03-25 00:32:19 +08:00
while true do
let result = Parser.main Lexer.token lexbuf in
2024-03-25 00:34:14 +08:00
Printf.printf "%d" (interp#interp result); print_newline(); flush stdout
2024-03-25 00:32:19 +08:00
done
with Lexer.Eof ->
exit 0