fp-course icon indicating copy to clipboard operation
fp-course copied to clipboard

Test for List Applicative instance is ambigious

Open mark-schultz-wu opened this issue 6 years ago • 1 comments

The test suite for Applicative.hs accepts the following code:

(<*>) (f :. fs) xs = (f <$> xs) ++ (fs <*> xs)
(<*>) _ _ = Nil

and rejects the following code:

(<*>) (f :. fs) (x :. xs) = (f x) :. (fs <*> xs)
(<*>) _ _ = Nil

I'm under the impression that these are both valid applicative instances (I think the second one is essentially the ZipList applicative instance, but I'm not actually sure). If so, it might be useful to somehow alert the students that, while the second instance is valid, they should search for an entirely different instance (and not stare at that one and try to get it to work).

mark-schultz-wu avatar May 13 '18 04:05 mark-schultz-wu

You are correct that the second instance is the ZipList Applicative. When a type has both a Monad and an Applicative instance, it's necessary for them to agree. That is, given ap,

ap :: Monad m => m (a -> b) -> m a -> m b
ap mf ma = do
  f <- mf
  a <- ma
  pure (f a)

then for any values, f `ap` a == f <*> a. Since ZipList does not have a Monad, we give the "cross-product-y" one to List itself.

Adding a comment to indicate two possible instances is a good idea :) Thanks

parsonsmatt avatar May 13 '18 05:05 parsonsmatt