过滤无限的monadic值列表

时间:2013-03-09 03:35:34

标签: haskell monads

也许这很明显,但我似乎无法弄清楚如何最好地过滤无限的IO值列表。这是一个简化的例子:

infinitelist :: [IO Int]

predicate :: (a -> Bool)

-- how to implement this?
mysteryFilter :: (a -> Bool) -> [IO a] -> IO [a]

-- or perhaps even this?
mysteryFilter' :: (a -> Bool) -> [IO a] -> [IO a]

也许我必须以某种方式使用sequence,但我希望评估是懒惰的。有什么建议?实质是,对于输出中的每个IO Int,我们可能必须检查输入中的多个IO Int值。

谢谢!

1 个答案:

答案 0 :(得分:11)

如果不使用unsafeInterleaveIO或类似的东西,就不可行。你不能用第二种类型的签名来编写过滤器,因为如果可以的话,你可以说

unsafePerformIOBool :: IO Bool -> Bool
unsafePerformIOBool m  = case mysteryFilter' id [m] of
    []    -> False
    (_:_) -> True

类似地,第一个类型的签名不起作用 - 任何递归调用都会返回类型为IO [a]的东西,但是为了构建一个列表,你需要执行在返回结果之前执行此操作(因为:不在IO中,您需要使用>>=)。通过归纳,您必须执行列表中的所有操作(在列表无限长时才会执行此操作),然后才能返回结果。

unsafeInterleaveIO解决了这个问题,但不安全。

 mysteryFilter f [] = return []
 mysteryFilter f (x:xs) = do ys <- unsafeInterleaveIO $ mysteryFilter f xs
                             y <- x
                             if f y then return (y:ys) else return ys

问题在于这打破了monad应该提供的顺序。你不再保证你的monadic动作何时发生(它们可能永远不会发生,它们可能会发生多次,等等)。

列表只是不喜欢IO。这就是我们拥有大量流媒体类型(Iteratees,Conduits,Pipes等)的原因。

最简单的类型可能是

data MList m a = Nil | Cons a (m (MList m a))

请注意我们观察到

[a] == MList Id a

因为

toMList :: [a] -> MList Id a
toMList [] = Nil
toMList (x:xs) = Cons x $ return $ toMList xs

fromMList :: MList Id a -> [a]
fromMList Nil = []
fromMList (Cons x xs) = x:(fromMList . runId $ xs)

另外,MList是一个仿函数

instance Functor m => Functor (MList m) where
  fmap f Nil = Nil
  fmap f (Cons x xs) = Cons (f x) (fmap (fmap f) xs)

它是Functor和Natural变换类别中的仿函数。

trans :: Functor m => (forall x. m x -> n x) -> MList m a -> MList n a
trans f Nil = Nil
trans f (Cons x xs) = Cons x (f (fmap trans f xs))

用这个很容易写出你想要的东西

mysteryFilter :: (a -> Bool) -> MList IO (IO a) -> IO (MList IO a)
mysteryFilter f Nil = return Nil
mysteryFilter f (Cons x xs)
  = do y <- x
       let ys = liftM (mysteryFilter f) xs
       if f y then Cons y ys else ys

或其他各种类似的功能。