-- (c) MP-I (1998/9-2006/7) and CP (2005/6-2014/5) module Cp where infix 5 >< infix 4 -|- -- (1) Product ----------------------------------------------------------------- split :: (a -> b) -> (a -> c) -> a -> (b,c) split f g x = (f x, g x) (><) :: (a -> b) -> (c -> d) -> (a,c) -> (b,d) f >< g = split (f . p1) (g . p2) -- the 0-adic split (!) :: a -> () (!) = const () -- Renamings: p1 = fst p2 = snd -- (2) Coproduct --------------------------------------------------------------- -- Renamings: i1 = Left i2 = Right -- either is predefined (-|-) :: (a -> b) -> (c -> d) -> Either a c -> Either b d f -|- g = either (i1 . f) (i2 . g) -- McCarthy's conditional: cond p f g = (either f g) . (grd p) -- (3) Exponentiation --------------------------------------------------------- -- curry is predefined ap :: (a -> b,a) -> b ap = uncurry ($) expn :: (b -> c) -> (a -> b) -> a -> c expn f = curry (f . ap) p2p p b = if b then (snd p) else (fst p) -- pair to predicate -- exponentiation functor is (a->) predefined -- instance Functor ((->) s) where -- fmap f g = f . g -- (4) Others ----------------------------------------------------------------- --const :: a -> b -> a st const a x = a is predefined grd :: (a -> Bool) -> a -> Either a a grd p x = if p x then Left x else Right x -- (5) Natural isomorphisms ---------------------------------------------------- swap :: (a,b) -> (b,a) swap = split snd fst assocr :: ((a,b),c) -> (a,(b,c)) assocr = split ( fst . fst ) (split ( snd . fst ) snd ) assocl :: (a,(b,c)) -> ((a,b),c) assocl = split ( id >< p1 ) ( p2 . p2 ) undistr :: Either (a,b) (a,c) -> (a,Either b c) undistr = either ( id >< i1 ) ( id >< i2 ) undistl :: Either (b, c) (a, c) -> (Either b a, c) undistl = either ( i1 >< id ) ( i2 >< id ) flatr :: (a,(b,c)) -> (a,b,c) flatr (a,(b,c)) = (a,b,c) flatl :: ((a,b),c) -> (a,b,c) flatl ((b,c),d) = (b,c,d) -- pwnil = split id (!) br = split id (!) -- bang on the right, old pwnil means "pair with nil" bl = swap . br coswap :: Either a b -> Either b a coswap = either i2 i1 coassocr :: Either (Either a b) c -> Either a (Either b c) coassocr = either (id -|- i1) (i2 . i2) coassocl :: Either b (Either a c) -> Either (Either b a) c coassocl = either (i1.i1) (i2 -|- id) distr = uncurry f where f a = (g a) -|- (g a) where g a x = (a,x) distl = (swap -|- swap) . distr . swap -- (6) Class bifunctor --------------------------------------------------------- class BiFunctor f where bmap :: (a -> b) -> (c -> d) -> (f a c -> f b d) instance BiFunctor Either where bmap f g = f -|- g instance BiFunctor (,) where bmap f g = f >< g -- (7) Monads: ----------------------------------------------------------------- -- (7.1) Kleisli monadic composition ------------------------------------------- infix 4 .! (.!) :: Monad a => (b -> a c) -> (d -> a b) -> d -> a c (f .! g) a = (g a) >>= f mult :: (Monad m) => m (m b) -> m b -- also known as join mult = (>>= id) -- (7.2) Monadic binding --------------------------------------------------------- ap' :: (Monad m) => (a -> m b, m a) -> m b ap' = uncurry (flip (>>=)) -- (7.3) Lists singl :: a -> [a] singl = return -- (7.4) Strong monads ----------------------------------------------------------- class (Functor f, Monad f) => Strong f where rstr :: (f a,b) -> f(a,b) rstr(x,b) = do a <- x ; return (a,b) lstr :: (b,f a) -> f(b,a) lstr(b,x) = do a <- x ; return (b,a) instance Strong IO instance Strong [] instance Strong Maybe dstr :: Strong m => (m a, m b) -> m (a, b) --- double strength --dstr = mult . fmap rstr . lstr dstr = rstr .! lstr splitm :: Strong ff => ff (a -> b) -> a -> ff b -- Exercise 4.8.13 in Jacobs' "Introduction to Coalgebra" (2012) splitm = curry (fmap ap . rstr) {-- -- (7.5) Monad transformers ------------------------------------------------------ class (Monad m, Monad (t m)) => MT t m where -- monad transformer class lift :: m a -> t m a -- nested lifting: dlift :: (MT t (t1 m), MT t1 m) => m a -> t (t1 m) a dlift = lift . lift --} -- (8) Basic functions, abbreviations ------------------------------------------ zero = const 0 one = const 1 nil = const [] cons = uncurry (:) add = uncurry (+) mul = uncurry (*) conc = uncurry (++) -- (9) Advanced ---------------------------------------------------------------- class (Functor f) => Unzipable f where unzp :: f(a,b) -> (f a,f b) unzp = split (fmap p1)(fmap p2) class Functor g => DistL g where lamb :: Monad m => g (m a) -> m (g a) instance DistL [] where lamb = sequence --------------------------------------------------------------------------------