[英]Haskell: How can I implement an applicative functor on own map data type?
我对 Haskell 很陌生,我在 Haskell 中编写了一个数据类型来表示区间映射。
这意味着什么? 简而言之:一种地图数据类型,它为每个可能的键返回一个值(在我的例子中简单地说是 [0..])。
然后你插入“序列”,就像我希望我的地图从 7 到 23 'b' 所以键 0 到 6 将是初始值,例如 'a' 和 7 到 23 将是 'b' 和 24 并且正在进行的将是 'a ' 再等等。
我设法编写了数据类型、获取和插入函数以及函子版本。
但是我无法使应用函子版本正常工作。 这个想法是将键值设置为 [0..] 并只处理这些值。
这是我的代码,感谢您提供的任何帮助!
-- Building an interval map data structure in haskell
data IntervalMap k v = IntervalMap {keys :: [k] , values :: [v]} | Empty deriving Show
-- k = key, Typ variable
-- v = value, Typ variable
singleton :: (Enum k, Num k) => v -> IntervalMap k v
singleton v = IntervalMap{keys=[0..], values= repeat v}
-- get operator => a ! 5 = value at position 5
(!) :: Ord k => IntervalMap k v -> k -> v
(!) iMap k = snd (head (filter (\(x, y) -> x == k) (zip (keys iMap) (values iMap)) ))
-- insert a sequence into intervalMap
insert :: (Ord k, Num k, Enum k) => k -> k -> v -> IntervalMap k v -> IntervalMap k v
insert start end value iMap = IntervalMap {keys=keys iMap, values = rangeChanger (values iMap) start end value}
-- helper function to change a range of values in an intervalMap
rangeChanger :: (Num a1, Enum a1, Ord a1) => [a2] -> a1 -> a1 -> a2 -> [a2]
rangeChanger iMapValues start end value = [if (i >= start) && (i <= end) then newValue else iMapValue | (iMapValue, newValue, i) <- zip3 iMapValues (repeat value) [0..]]
-- functor instance for intervalMap
instance Functor (IntervalMap k) where
-- fmap :: (a -> b) -> f a -> f b
fmap f iMap = IntervalMap {keys=keys iMap, values= map f (values iMap) }
-- applicative functor for intervalMap
instance (Ord k, Num k, Enum k) => Applicative (IntervalMap k) where
pure k = IntervalMap{keys=[0..], values=repeat k}
_ <*> Nothing = Nothing
-- HOW TO DO?
-- class Functor functor => Applicative functor where
-- pure :: a -> functor a
-- (<*>) :: functor (a -> b) -> functor a -> functor b
-- (*>) :: functor a -> functor b -> functor b
-- (<*) :: functor a -> functor b -> functor a
似乎您总是希望键为[0..]
,例如它在您的rangeChanger
函数中进行了硬编码。 如果是这样,那么它是多余的,老实说,我会把它排除在外。 您可以像在rangeChanger
函数中那样通过执行zip [0..] (values iMap)
类的操作轻松地重建它。
如果您进行了更改,那么您的IntervalMap
数据结构基本上与ZipList
相同,后者在此处具有一个应用实例:
instance Applicative ZipList where pure x = ZipList (repeat x) liftA2 f (ZipList xs) (ZipList ys) = ZipList (zipWith f xs ys)
您会看到这没有定义<*>
但可以根据liftA2
来定义: p <*> q = liftA2 (\\fx -> fx) pq
,因此您也可以为ZipList
显式编写它:
ZipList fs <*> ZipList xs = ZipList (zipWith (\f x -> f x) fs xs)
编辑:我还应该提到与ZipList
一个区别是您的IntervalMap
类型有一个Empty
构造函数。 这让事情变得更难,您需要知道您的值具有某种默认值,但这通常是不可能的(并非每个类型都有默认值),因此您的类型不能是 Applicative。 你真的需要那个Empty
箱子吗?
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.