forked from panesofglass/StateMonad
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathMumble004.lhs
53 lines (40 loc) · 1.34 KB
/
Mumble004.lhs
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
> data Tr a = Lf a | Br [Tr a]
> deriving Show
> tr0 = Br [Lf 'a', Br [Lf 'b', Lf 'c']]
> tr1 = Br [Lf 'a', Br [Br [Lf 'b', Lf 'c'], Lf 'd']]
> type LbTr a = (Tr (St, a))
> type St = Int
> label :: Tr a -> LbTr a
> label tr = snd (lab tr 0) where
> lab :: Tr a -> St -> (St, LbTr a)
> lab (Lf contents) n = ((n+1), (Lf (n, contents)))
> lab (Br trs) n0 = let l = thd trs n0 []
> in (fst (last l), Br (map snd l))
> where
> thd :: [Tr a] -> St -> [(St, LbTr a)] -> [(St, LbTr a)]
> thd [] _ acc = reverse acc
> thd (t:trs) n0 acc = let (n1, nt1) = lab t n0
> in thd trs n1 ((n1, nt1):acc)
> newtype Labeled a = Labeled (St -> (St, a))
> instance Monad Labeled where
> return contents = Labeled (\st -> (st, contents))
> Labeled fst0 >>= fany1 =
> Labeled $ \st0 ->
> let (st1, any1) = fst0 st0
> Labeled fst1 = fany1 any1
> in fst1 st1
> mlabel :: Tr a -> LbTr a
> mlabel tr = let Labeled mt = insert tr
> in snd (mt 0)
> insert :: Tr a -> Labeled (LbTr a)
> insert (Lf x)
> = do n <- updateState
> return $ Lf (n,x)
> insert (Br trs)
> = do mtrs <- mapM insert trs
> return $ Br mtrs
> updateState :: Labeled St
> updateState = Labeled (\n -> ((n+1),n))
> t1 = 5
> f x = t1 + x
> main = print $ mlabel tr1