Scrap your boilerplate: Difference between revisions
(link to generics papers) |
m (→fmap: typo) |
||
(One intermediate revision by the same user not shown) | |||
Line 26: | Line 26: | ||
This is most likely traverses more than a purpose-written Functor instance. | This is most likely traverses more than a purpose-written Functor instance. | ||
It ''is'' possible to | It ''is'' possible to modify parameterized types: | ||
<haskell> | <haskell> | ||
Line 61: | Line 61: | ||
</haskell> | </haskell> | ||
Compare this with an approach that avoids the unsafeCoerce, at the cost of additionally requiring an implementation of 'gunfold': [http://code.haskell.org/generics/comparison/SYB1_2/GMap.lhs GMap.lhs]. |
Latest revision as of 01:33, 28 December 2013
Scrap Your Boilerplate (syb) is an infrastructure for generic programming in Haskell. (This should not be confused with the concept of "generic types" in Java, which in the Java context means parameterized types which allow you to specify the type of list elements and such things - a feature Haskell has had since its inception.)
In the current GHC implementation, it consists of a set of modules (Data.Generics and its submodules) and deriving support for the Data and Typeable classes from those modules.
Other examples using syb and other libraries can be found Research papers/Generics
Code snippets
ListifyWholeLists
This code snippet implements a variation on the listify function. listify recurses into every "node" in a datastructure, which is not so good if what you're searching for is lists (e.g. strings) and you don't want to pick up sublists (substrings, in the case of strings). This function does not recurse into lists of the type being searched for, avoiding that problem.
It illustrates the use of ShowS-style function composition as an alternative to recursive list concatenation - the latter is inefficient. It also illustrates how laziness automatically means that sublists of type [b] won't be examined, without any special effort required. Isn't laziness wonderful?
import Data.Generics
-- A version of Data.Generics.listify which doesn't recurse into sublists of type [b]
listifyWholeLists :: Typeable b => ([b] -> Bool) -> GenericQ [[b]]
listifyWholeLists blp = flip (synthesize id (.) (mkQ id (\bl _ -> if blp bl then (bl:) else id))) []
fmap
This is most likely traverses more than a purpose-written Functor instance.
It is possible to modify parameterized types:
{-# LANGUAGE DeriveDataTypeable, ScopedTypeVariables, FlexibleContexts #-}
import Data.Generics
import Unsafe.Coerce
{- | C tags the type that is actually parameterized, so to avoid touching the
Int when a ~ Int:
> data T a = T Int a
by changing the type (not representation) to:
> x :: T Int (C Int)
-}
newtype C a = C a deriving (Data,Typeable)
fmapData :: forall t a b. (Typeable a, Data (t (C a)), Data (t a)) =>
(a -> b) -> t a -> t b
fmapData f input = uc . everywhere (mkT $ \(x::C a) -> uc (f (uc x)))
$ (uc input :: t (C a))
where uc = unsafeCoerce
Example (in case you don't believe me)
Main*> fmapData succ (T 1 1 :: T Int)
T 1 2
Main*> fmapData succ [1,2,3::Int]
[2,3,4]
Compare this with an approach that avoids the unsafeCoerce, at the cost of additionally requiring an implementation of 'gunfold': GMap.lhs.