[Haskell-cafe] Rationale for two separate map functions in ClassyPrelude
adam vogt
vogt.adam at gmail.com
Sat Feb 14 07:10:59 UTC 2015
Well Unboxed.Vector can have an instance of CanMap if we use
-XOverlappingInstances, (see
https://gist.github.com/aavogt/f64db5678181307bebb9). Unfortunately I
had to use functional dependencies, since associated type families
can't overlap.
On Sat, Feb 14, 2015 at 12:07 AM, Amos Robinson <amos.robinson at gmail.com> wrote:
> Sorry, I just reread the user guide and it makes sense. That's a shame -
> this had me excited.
>
> On Sat Feb 14 2015 at 3:52:12 PM Amos Robinson <amos.robinson at gmail.com>
> wrote:
>>
>> I feel a bit silly, but could you explain to me where the overlap is here?
>>
>> On Sat Feb 14 2015 at 3:45:30 PM adam vogt <vogt.adam at gmail.com> wrote:
>>>
>>> Hi Clinton,
>>>
>>> I think the problem is that the instance:
>>>
>>> > instance (Functor f) => CanMap (f a) b
>>>
>>> is overlapped by instances for containers that take a parameter but
>>> are not instances of Functor. For example:
>>>
>>> > instance (Unbox a, Unbox b) => CanMap (Data.Vector.Unboxed.Vector a) b
>>>
>>> Using {-# LANGUAGE OverlappingInstances #-} is a big deal for some
>>> people.
>>>
>>> Regards,
>>> Adam
>>>
>>> On Wed, Feb 11, 2015 at 2:00 PM, Clinton Mead <clintonmead at gmail.com>
>>> wrote:
>>> > ClassyPrelude has two map functions, namely:
>>> >
>>> > 1. "map"
>>> > 2. "omap"
>>> >
>>> > "map" works on any Functor. However, things like "Text" are not
>>> > functors as
>>> > they aren't generic containers. As can be seen in the following code:
>>> >
>>> > module Main where
>>> > import Prelude ()
>>> > import ClassyPrelude
>>> > import qualified Data.Text as T
>>> > import Data.Char as C
>>> >
>>> > main = do
>>> > let l = [1,2,3] :: [Int]
>>> > let t = (T.pack "Hello")
>>> > let m = Just 5
>>> > print $ map (*2) l
>>> > print $ map (*2) m
>>> > print $ omap C.toUpper t
>>> > return ()
>>> >
>>> >
>>> > Notice one has to use "omap" to deal with the Text. The thing is, I
>>> > found it
>>> > trivially easy to get "map" to work for both calls. Here's the code:
>>> >
>>> > {-# LANGUAGE MultiParamTypeClasses #-}
>>> > {-# LANGUAGE TypeFamilies #-}
>>> >
>>> > module Main where
>>> > import Prelude hiding (map)
>>> > import qualified Data.Text as T
>>> > import Data.Char as C
>>> > import Control.Monad (Functor)
>>> >
>>> > class CanMap a b where
>>> > type Element a :: *
>>> > type Container a b :: *
>>> > map :: (Element a -> b) -> a -> Container a b
>>> >
>>> > instance (Functor f) => CanMap (f a) b where
>>> > type Element (f a) = a
>>> > type Container (f a) b = f b
>>> > map = fmap
>>> >
>>> > instance CanMap T.Text Char where
>>> > type Element T.Text = Char
>>> > type Container T.Text Char = T.Text
>>> > map = T.map
>>> >
>>> > main = do
>>> > let l = [1,2,3] :: [Int]
>>> > let m = Just 5
>>> > let t = (T.pack "Hello")
>>> > print $ map (*2) l
>>> > print $ map (*2) m
>>> > print $ map C.toUpper t
>>> > return ()
>>> >
>>> >
>>> > All that's required is to add instances to CanMap for any monomorphic
>>> > containers. ClassyPrelude already does this anyway with "omap" in the
>>> > Data.MonoTraversable module. I suspect however there's a good reason
>>> > I'm
>>> > missing about why there should be two separate map functions to deal
>>> > with
>>> > these alternate situations, but I'm wondering what that is.
>>> >
>>> > _______________________________________________
>>> > Haskell-Cafe mailing list
>>> > Haskell-Cafe at haskell.org
>>> > http://www.haskell.org/mailman/listinfo/haskell-cafe
>>> >
>>> _______________________________________________
>>> Haskell-Cafe mailing list
>>> Haskell-Cafe at haskell.org
>>> http://www.haskell.org/mailman/listinfo/haskell-cafe
More information about the Haskell-Cafe
mailing list