I'm writing a function that does some searching in a sequence of arbitrary symbols. I'd like to make it generic enough so that it works on lists, Foldable
s as well on ByteString
s and Text
s. Generalizing it to Foldable
is simple. But how to include ByteString
s and Text
s? Sure I could convert ByteString
into a list and then call my function, but I'd lose all the advantages ByteString
s.
To have a concrete example let's say we want to make a histogram function:
import Control.Monad.State
import qualified Data.Foldable as F
import Data.Map.Strict (Map)
import qualified Data.Map.Strict as Map
import Data.Word
import qualified Data.ByteString as B
import qualified Data.Text as T
type Histogram a = Map a Int
empty :: (Ord a) => Histogram a
empty = Map.empty
histogramStep :: (Ord a) => a -> Histogram a -> Histogram a
histogramStep k = Map.insertWith (+) k 1
histogram :: (Ord a, F.Foldable t) => t a -> Histogram a
histogram = F.foldl (flip histogramStep) empty
But since neither ByteString
nor Text can be Foldable
(it stores just Word8
s/Char
s, not arbitrary elements), I'm stuck with creating more functions that look exactly like the one before, just with a different type signatures:
histogramBS :: B.ByteString -> Histogram Word8
histogramBS = B.foldl (flip histogramStep) empty
histogramText :: T.Text -> Histogram Char
histogramText = T.foldl (flip histogramStep) empty
This is something one does not expect in a functional language like Haskell.
How to make it generic, to write histogram
once and for all?
We also discussed some uncommonly used list built-in functions like any () and all (). For each function, we demonstrated its usage and saw how it applies on lists with examples.
The list created with this method is immutable as well, so you are sure that there will not be any more elements in list, at any condition. For Example, you can use this list as follows. This method can also help in creating the List quickly, but created list is not immutable.
This is due to a list’s characteristic called mutability, which enables us to modify the lists directly. We have functions that are commonly used to manipulate lists. For example: len (), sum (), max (), range () and many more. We also have some functions that are not commonly used like any (), all (), etc.
Given below are some important Python list built-in functions. Kindly visit the Python official documentation page for details of these functions. Returns the number of element in the list . Creates a list out of an iterable. Returns an iterator of integers from start to stop, with an increment of step. Adds all items of an iterable.
Your solution is pretty much what the ListLike package does. There's also the additional package listlike-instances which adds instances for Text
and Vector
.
After a while I made a solution myself, but I'm not sure if it could be solved in a better way, or if someone already did this in some library.
I created a type-class with TypeFamilies
as
class Foldable' t where
type Element t :: *
foldlE :: (b -> Element t -> b) -> b -> t -> b
-- other functions could be copied here from Foldable
and instances:
newtype WrapFoldable f a = WrapFoldable { unwrapFoldable :: f a }
instance (F.Foldable f) => Foldable' (WrapFoldable f a) where
type Element (WrapFoldable f a) = a
foldlE f z = F.foldl f z . unwrapFoldable
instance Foldable' B.ByteString where
type Element B.ByteString = Word8
foldlE = B.foldl
instance Foldable' T.Text where
type Element (T.Text) = Char
foldlE = T.foldl
or even better with FlexibleInstances
:
instance (F.Foldable t) => Foldable' (t a) where
type Element (t a) = a
foldlE = F.foldl
Now I can write (with FlexibleContexts
):
histogram :: (Ord (Element t), Foldable' t) => t -> Histogram (Element t)
histogram = foldlE (flip histogramStep) empty
and use it on Foldable
s, ByteString
s, Text
s etc.
You might consider objectifying folds themselves:
{-# LANGUAGE GADTs #-}
import Data.List (foldl', unfoldr)
import qualified Data.ByteString.Lazy as B
import qualified Data.Vector.Unboxed as V
import qualified Data.Text as T
import qualified Data.Map as Map
import Data.Word
type Histogram a = Map.Map a Int
empty :: (Ord a) => Histogram a
empty = Map.empty
histogramStep :: (Ord a) => Histogram a -> a -> Histogram a
histogramStep h k = Map.insertWith (+) k 1 h
histogram :: Ord b => Fold b (Histogram b)
histogram = Fold histogramStep empty id
histogramT :: T.Text -> Histogram Char
histogramT = foldT histogram
histogramB :: B.ByteString -> Histogram Word8
histogramB = foldB histogram
histogramL :: Ord b => [b] -> Histogram b
histogramL = foldL histogram
-- helper library
-- see http://squing.blogspot.fr/2008/11/beautiful-folding.html
-- note existential type
data Fold b c where Fold :: (a -> b -> a) -> !a -> (a -> c) -> Fold b c
instance Functor (Fold b) where fmap f (Fold op x g) = Fold op x (f . g)
foldL :: Fold b c -> [b] -> c
foldL (Fold f x c) bs = c $ (foldl' f x bs)
foldV :: V.Unbox b => Fold b c -> V.Vector b -> c
foldV (Fold f x c) bs = c $ (V.foldl' f x bs)
foldT :: Fold Char t -> T.Text -> t
foldT (Fold f x c) t = c $ (T.foldl' f x t)
foldB :: Fold Word8 t -> B.ByteString -> t
foldB (Fold f x c) t = c $ (B.foldl' f x t)
sum_, product_ :: Num a => Fold a a
sum_ = Fold (+) 0 id
product_ = Fold (*) 1 id
length_ :: Fold a Int
length_ = Fold (const . (+1)) 0 id
maximum_ = Fold max 0 id
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With