-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathHW4-AlessioBandiera.hs
303 lines (267 loc) · 13.9 KB
/
HW4-AlessioBandiera.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
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
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
import Data.Char (toLower)
import Control.Monad.State
import Data.Maybe (fromJust)
import Control.Applicative
-- ### Esercizio 1
count :: Eq a => a -> [a] -> Int
count x = length . filter (x==)
getStrings :: Int -> IO [String]
getStrings 0 = return []
getStrings n = do ls <- getStrings (n - 1)
l <- getLine
return (l : ls)
contains :: Char -> String -> Bool
contains c = any (\x -> x == c)
done :: IO ()
done = return ()
charCount :: IO ()
charCount = do putStrLn "Insert a positive integer: "
input <- getLine
let n = read input :: Int
putStrLn ("Insert " ++ show n ++ " strings:")
strings <- getStrings n
let lowerStrings = map (map toLower) strings
putStrLn "These are the occurrences of each letter (case-insensitive):"
foldr (>>) done (map (\c -> putStrLn (c : ": " ++ show (sum (map (fromEnum . contains c) lowerStrings)))) ['a'..'z'])
-- ### Esercizio 2
data BinTree a = Node a (BinTree a) (BinTree a) | Empty
deriving Show
-- Per risolvere questo esercizio con le monadi sarebbe stato sufficiente
-- avere come stato una tupla di interi per mantenere ad esempio il valore
-- del cammino dalla radice al nodo corrente (meno il nodo corrente), e
-- il totale del sottoalbero corrente (compreso il nodo corrente). Il problema
-- di questa soluzione è che, per attuarla, è necessario salvare dei valori
-- intermedi, dai quali le successive istruzioni dipendono: ad esempio,
-- è strettamente necessario salvarsi il valore del cammino dalla radice
-- al nodo corrente *prima di fare entrambe le chiamate ricorsive*, e
-- soprattutto, dopo aver fatto la prima chiamata ricorsiva, di riaggiornare
-- il valore dello stato usato per salvare il cammino dalla radice con il
-- valore preso *prima della prima ricorsione*, poiché dopo la prima ricorsione
-- lo stato conterrà un valore errato del cammino dalla radice ("sporcato"
-- dalla ricorsione stessa). Questo non è un problema per quanto riguarda il
-- codice con le monadi, poiché per come queste funzionano non si crea nessun
-- problema nel riutilizzare valori intermedi; il problema si verifica nel momento
-- in cui tale codice deve essere scritto con Applicative soltanto. Infatti,
-- proprio perché quest'idea non può essere scritta senza creare questo tipo
-- di dipendenze, non è stato possibile scrivere questo algoritmo, ma l'algoritmo
-- seguente permette invece di risolvere l'esercizio senza creare dipendenze:
-- - lo stato dell'algoritmo sarà un (Int, [Int]), dove il primo intero
-- rappresenta il cammino dalla radice al nodo corrente (nodo corrente
-- escluso), mentre la lista rappresenta lo "stack dei totali" dei sottoalberi
-- - salvare il valore del cammino dalla radice (che esclude il nodo corrente),
-- ed ad esso sommargli il valore del nodo corrente (per le chiamate successive)
-- - eseguire la ricorsione sul figlio sinistro (per la quale era necessario
-- aggiungere il valore del nodo corrente all'interno dello stato)
-- - eseguire la ricorsione sul figlio destro (discorso analogo)
-- - sia `totSum` la somma tra l'etichetta del nodo corrente e i primi due elementi
-- della lista salvata nello stato; allora, è necessario salvarsi `totSum` per
-- controllare se il nodo è bilanciato, e contemporaneamente
-- - decrementare il valore del cammino dalla radice (poiché per ritornare al
-- padre il valore del nodo corrente non deve essere incluso, esso serviva
-- solo per effettuare le chiamate ricorsive sui figli)
-- - rimpiazzare i primi due elementi della lista nello stato con `totSum`,
-- poiché questo rappresenta di fatto il totale del sottoalbero corrente,
-- nodo corrente incluso, da rimandare al padre (infatti le foglie non
-- fanno altro che aggiungere 0 in testa alla lista corrente)
-- Questo algoritmo, poiché mantiene lo "stack dei totali" dei sottoalberi, non
-- necessita di generare dipendenze del tipo descritto precedentemente, ed è
-- dunque possibile trasformare il codice scritto con le monadi in un codice
-- che utilizza esclusivamente Applicative.
append x = state (\(n, l) -> (n, (n, x : l)))
fresh a = state (\(n, l) -> (n, (n + a, l)))
updateList a = state (\(n, l) -> let (fs, rs) = splitAt 2 l
totSum = a + sum fs
in (totSum, (n - a, totSum : rs)))
updateNodes n resSx resDx totSum a = (if n == totSum then (a:) else id) resSx ++ resDx
balancedNodesM :: BinTree Int -> [Int]
balancedNodesM b = evalState (balancedNodesMAux b) (0, [])
where
balancedNodesMAux Empty = do _ <- append 0
return []
balancedNodesMAux (Node a sx dx) = do n <- fresh a
resSx <- balancedNodesMAux sx
resDx <- balancedNodesMAux dx
totSum <- updateList a
return (updateNodes n resSx resDx totSum a)
balancedNodesA :: BinTree Int -> [Int]
balancedNodesA b = evalState (balancedNodesAAux b) (0, [])
where
balancedNodesAAux Empty = append 0 *> pure []
balancedNodesAAux (Node a sx dx) = updateNodes
<$> fresh a
<*> balancedNodesAAux sx
<*> balancedNodesAAux dx
<*> updateList a
<*> pure a
-- ### Esercizio 3
-- il costruttore più esterno è il LSB
data NatBin = End | Zero NatBin | One NatBin
deriving Show
instance Eq NatBin where
m == n = equalAux (removeLeadingZeros m) (removeLeadingZeros n)
where
equalAux End End = True
equalAux End _ = False
equalAux _ End = False
equalAux (Zero x) (Zero y) = equalAux x y
equalAux (Zero x) (One y) = False
equalAux (One x) (Zero y) = False
equalAux (One x) (One y) = equalAux x y
instance Ord NatBin where
m <= n = let (c, t) = compareAux (removeLeadingZeros m) (removeLeadingZeros n) in if t then c else True
where
compareAux End End = (True, False)
compareAux _ End = (False, True)
compareAux End _ = (True, True)
compareAux (Zero x) (Zero y) = compareAux x y
compareAux (Zero x) (One y) = let (c, t) = compareAux x y in if t then (c, t) else (True, True)
compareAux (One x) (Zero y) = let (c, t) = compareAux x y in if t then (c, t) else (False, True)
compareAux (One x) (One y) = compareAux x y
isByte :: (Ord a, Num a) => a -> Bool
isByte x = (x <= 255) && (x >= 0)
fromNatBin :: NatBin -> Maybe Int
fromNatBin x = let n = fromNatBinAux x 0 in if isByte n then Just n else Nothing
where
fromNatBinAux (Zero x) i = fromNatBinAux x (i + 1)
fromNatBinAux (One x) i = 1 * (2^i) + fromNatBinAux x (i + 1)
fromNatBinAux End _ = 0
intoNatBin :: Int -> Maybe NatBin
intoNatBin 0 = Just (Zero End)
intoNatBin n = if isByte n then Just (intoNatBinAux (n `divMod` 2)) else Nothing
where
intoNatBinAux (0, 0) = End
intoNatBinAux (n, 0) = Zero (intoNatBinAux (n `divMod` 2))
intoNatBinAux (n, 1) = One (intoNatBinAux (n `divMod` 2))
removeLeadingZeros :: NatBin -> NatBin
removeLeadingZeros n = fst (removeLeadingZerosAux n 0 0)
where
removeLeadingZerosAux End c l = if l == c then (End, c - 1) else (End, c)
removeLeadingZerosAux (Zero x) c l = let (res, c') = removeLeadingZerosAux x (c + 1) (l + 1) in if c' /= 0 then (res, c' - 1) else (Zero res, c')
removeLeadingZerosAux (One x) c l = let (res, c') = removeLeadingZerosAux x 0 (l + 1) in (One res, 0)
zeroNatBin = Zero End
oneNatBin = One End
minValueNatBin = zeroNatBin
maxValueNatBin = fromJust (intoNatBin 255)
lengthNatBin :: NatBin -> Int
lengthNatBin m = lengthNatBinAux (removeLeadingZeros m)
where
lengthNatBinAux End = 0
lengthNatBinAux (Zero x) = 1 + lengthNatBinAux x
lengthNatBinAux (One x) = 1 + lengthNatBinAux x
isByteNatBin :: NatBin -> Bool
isByteNatBin n = n <= maxValueNatBin
isEndNatBin :: NatBin -> Bool
isEndNatBin n = n == End
data Term = Value NatBin | Add Term Term | Sub Term Term | Mul Term Term | Div Term Term | Mod Term Term
deriving Show
data MaybeTerm a = JustTerm a | ZeroDivisionErr | NegativeNumberErr | OverflowErr | InvalidNatBinErr
deriving Show
instance Functor MaybeTerm where
fmap f ZeroDivisionErr = ZeroDivisionErr
fmap f NegativeNumberErr = NegativeNumberErr
fmap f OverflowErr = OverflowErr
fmap f InvalidNatBinErr = InvalidNatBinErr
fmap f (JustTerm x) = JustTerm (f x)
instance Applicative MaybeTerm where
pure = JustTerm
ZeroDivisionErr <*> _ = ZeroDivisionErr
NegativeNumberErr <*> _ = NegativeNumberErr
OverflowErr <*> _ = OverflowErr
InvalidNatBinErr <*> _ = InvalidNatBinErr
(JustTerm f) <*> mx = fmap f mx
instance Monad MaybeTerm where
return = pure
ZeroDivisionErr >>= _ = ZeroDivisionErr
NegativeNumberErr >>= _ = NegativeNumberErr
OverflowErr >>= _ = OverflowErr
InvalidNatBinErr >>= _ = InvalidNatBinErr
(JustTerm x) >>= f = f x
fromJustTerm :: MaybeTerm NatBin -> NatBin
fromJustTerm ZeroDivisionErr = error "MaybeTerm.fromJustTerm: ZeroDivisionErr"
fromJustTerm NegativeNumberErr = error "MaybeTerm.fromJustTerm: NegativeNumberErr"
fromJustTerm OverflowErr = error "MaybeTerm.fromJustTerm: OverflowErr"
fromJustTerm InvalidNatBinErr = error "MaybeTerm.fromJustTerm: InvalidNatBinErr"
fromJustTerm (JustTerm x) = x
checkNatBinValue :: NatBin -> MaybeTerm NatBin
checkNatBinValue n
| isEndNatBin n = InvalidNatBinErr
| not (isByteNatBin n) = OverflowErr
| otherwise = JustTerm (removeLeadingZeros n)
addNatBin :: NatBin -> NatBin -> MaybeTerm NatBin
addNatBin m n = let res = addNatBinAux m n 0 in if isByteNatBin res then JustTerm res else OverflowErr
where
addNatBinAux End End 0 = End
addNatBinAux End End 1 = One End
addNatBinAux (Zero x) End 0 = Zero (addNatBinAux x End 0)
addNatBinAux (Zero x) End 1 = One (addNatBinAux x End 0)
addNatBinAux (One x) End 0 = One (addNatBinAux x End 0)
addNatBinAux (One x) End 1 = Zero (addNatBinAux x End 1)
addNatBinAux End (Zero x) 0 = Zero (addNatBinAux End x 0)
addNatBinAux End (Zero x) 1 = One (addNatBinAux End x 0)
addNatBinAux End (One x) 0 = One (addNatBinAux End x 0)
addNatBinAux End (One x) 1 = Zero (addNatBinAux End x 1)
addNatBinAux (Zero x) (Zero y) 0 = Zero (addNatBinAux x y 0)
addNatBinAux (Zero x) (Zero y) 1 = One (addNatBinAux x y 0)
addNatBinAux (Zero x) (One y) 0 = One (addNatBinAux x y 0)
addNatBinAux (Zero x) (One y) 1 = Zero (addNatBinAux x y 1)
addNatBinAux (One x) (Zero y) 0 = One (addNatBinAux x y 0)
addNatBinAux (One x) (Zero y) 1 = Zero (addNatBinAux x y 1)
addNatBinAux (One x) (One y) 0 = Zero (addNatBinAux x y 1)
addNatBinAux (One x) (One y) 1 = One (addNatBinAux x y 1)
subNatBin :: NatBin -> NatBin -> MaybeTerm NatBin
subNatBin m n = if m >= n then JustTerm (subNatBinAux m n 0) else NegativeNumberErr
where
subNatBinAux End End _ = End
subNatBinAux (Zero x) End 0 = Zero (subNatBinAux x End 0)
subNatBinAux (Zero x) End 1 = One (subNatBinAux x End 1)
subNatBinAux (One x) End 0 = One (subNatBinAux x End 0)
subNatBinAux (One x) End 1 = Zero (subNatBinAux x End 0)
subNatBinAux End (Zero x) 0 = Zero (subNatBinAux End x 0)
subNatBinAux End (Zero x) 1 = One (subNatBinAux End x 1)
subNatBinAux End (One x) 0 = One (subNatBinAux End x 1)
subNatBinAux End (One x) 1 = Zero (subNatBinAux End x 1)
subNatBinAux (Zero x) (Zero y) 0 = Zero (subNatBinAux x y 0)
subNatBinAux (Zero x) (Zero y) 1 = One (subNatBinAux x y 1)
subNatBinAux (Zero x) (One y) 0 = One (subNatBinAux x y 1)
subNatBinAux (Zero x) (One y) 1 = Zero (subNatBinAux x y 1)
subNatBinAux (One x) (Zero y) 0 = One (subNatBinAux x y 0)
subNatBinAux (One x) (Zero y) 1 = Zero (subNatBinAux x y 0)
subNatBinAux (One x) (One y) 0 = Zero (subNatBinAux x y 0)
subNatBinAux (One x) (One y) 1 = One (subNatBinAux x y 1)
mulNatBin :: NatBin -> NatBin -> MaybeTerm NatBin
mulNatBin m n = mulNatBinAux m n
where
mulNatBinAux m n
| n == zeroNatBin = JustTerm zeroNatBin
| otherwise = do nMinusOne <- n `subNatBin` oneNatBin
res <- mulNatBinAux m nMinusOne
m `addNatBin` res
divModNatBin :: NatBin -> NatBin -> MaybeTerm (NatBin, NatBin)
divModNatBin m n = if n /= zeroNatBin then divModNatBinAux m n zeroNatBin else ZeroDivisionErr
where
divModNatBinAux m n q
| m < n = JustTerm (q, m)
| otherwise = do mMinusn <- m `subNatBin` n
qPlusOne <- q `addNatBin` oneNatBin
divModNatBinAux mMinusn n qPlusOne
evalTerm :: Term -> MaybeTerm NatBin
evalTerm (Value x) = checkNatBinValue x
evalTerm (Add x y) = do m <- evalTerm x
n <- evalTerm y
m `addNatBin` n
evalTerm (Sub x y) = do m <- evalTerm x
n <- evalTerm y
m `subNatBin` n
evalTerm (Mul x y) = do m <- evalTerm x
n <- evalTerm y
m `mulNatBin` n
evalTerm (Div x y) = do m <- evalTerm x
n <- evalTerm y
(q, _) <- m `divModNatBin` n
return q
evalTerm (Mod x y) = do m <- evalTerm x
n <- evalTerm y
(_, r) <- m `divModNatBin` n
return r
main :: IO ()
main = do putStrLn $ show $ "Alessio Bandiera 1985878"