-
Notifications
You must be signed in to change notification settings - Fork 23
/
Copy pathlab12-partial.hs
98 lines (80 loc) · 2.17 KB
/
lab12-partial.hs
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
import Data.Maybe
--- Limbajul și Interpretorul
-- Prescurtare pentru monada folosită
type M = Maybe
showM :: Show a => M a -> String
showM (Just x) = show x
showM Nothing = "<wrong>"
type Name = String
data Term
= Var Name
| Con Integer
| Term :+: Term
| Lam Name Term
| App Term Term
deriving (Show)
data Value
= Num Integer
| Fun (Value -> M Value)
instance Show Value where
show (Num x) = show x
show (Fun _) = "<function>"
-- Calculează rezultatul adunării a două valori.
-- Funcționează doar pentru numere.
add :: Value -> Value -> M Value
add (Num a) (Num b) = return $ Num (a + b)
add _ _ = Nothing
-- Calculează rezultatul aplicării unei funcții la o valoare.
-- Primul parametru trebuie să fie funcție.
apply :: Value -> Value -> M Value
apply (Fun f) v = f v
apply _ _ = Nothing
type Environment = [(Name, Value)]
interp :: Term -> Environment -> M Value
-- Încercăm să returnăm valoarea pentru variabila dată.
-- Dacă nu există, returnăm `Nothing`
interp (Var name) env = lookup name env
-- Returnăm valoarea pentru constanta dată
interp (Con i) _ = return $ Num i
-- Evaluăm suma a doi termeni
interp (t1 :+: t2) env = do
-- Evaluăm cei doi termeni
v1 <- interp t1 env
v2 <- interp t2 env
-- Calculăm suma valorilor lor
add v1 v2
interp (Lam paramName t) env = do
-- Definim o nouă funcție, care evaluează termenul într-un environment
-- în care avem o valoare dată pentru parametru
let lambdaFunc = \x -> interp t ((paramName, x) : env)
return $ Fun lambdaFunc
-- Aplicăm cei doi termeni
interp (App t1 t2) env = do
-- Le evaluăm valorile
v1 <- interp t1 env
v2 <- interp t2 env
-- Aplicăm primul termen cu al doilea
apply v1 v2
test :: Term -> String
test t = showM $ interp t []
-- Ar trebui să afișeze 7
pgm :: Term
pgm =
App (Lam "y" (
App (App
(Lam "f" (Lam "y" (App (Var "f") (Var "y"))))
(Lam "x" (Var "x" :+: Var "y")))
(Con 3))
) (Con 4)
-- Ar trebui să afișeze 42
pgm1 :: Term
pgm1 =
App
(Lam "x" ((Var "x") :+: (Var "x")))
((Con 10) :+: (Con 11))
-- Ar trebui să dea eroare
pgm2 :: Term
pgm2 = App (Con 2) (Con 3)
-- test pgm
-- test pgm1
-- test pgm2