-- (c) MP-I, CP, IPM (1998/99-2009/10) 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 :: a -> (a,()) -- pwnil means "pair with nil" pwnil = split id (!) 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) -- (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) 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 mult = (>>= id) -- (8) 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 [] {-- where rstr(l,x) = [ (a,x) | a <- l] lstr(x,l) = [ (x,a) | a <- l] --} instance Strong Maybe {-- where rstr(Nothing,l) = Nothing rstr(Just l,r) = Just(l,r) lstr(l,Nothing) = Nothing lstr(l,Just r) = Just(l,r) --} {-- (9) 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 --}