一个更严格的Control.Monad.Trans.Writer.Strict

时间:2014-09-09 15:56:21

标签: haskell lazy-evaluation monad-transformers strictness

所以我们有:

import Control.Monad.Writer.Strict

type M a = Writer (Map Key Val) a
某些KeyVal

只要我们不查看收集的输出,一切正常:

report comp = do
  let (a,w) = runWriter comp
  putStrLn a

但是如果我们想要检查w,我们会得到堆栈溢出。

 report comp = do
   let (a,w) = runWriter comp
   guard (not $ null w) $ do -- forcing w causes a stack overflow
     reportOutputs w
   putStrLn a

我认为原因是因为(>>=) is defined as Writer

m >>= k  = WriterT $ do
    (a, w)  <- runWriterT m
    (b, w') <- runWriterT (k a)
    return (b, w `mappend` w')

如果我有一个很大的Writer a计算,它会构建一长串的mappends:w <> (w' <> (w'' <> ...)),在这种情况下,它是Map.union,它在地图的脊柱中是严格的。因此,如果我构建了一个大的联合序列,那么只要我强制Map溢出堆栈,就必须对整个事物进行评估。

我们想要的是尽早完成工会。我们想要一个更严格的Strict.Writer:

m >>= k = WriterT $ do
    (a, w) <- runWriterT m
    (b, w') <- runWriterT (k a)
    let w'' = w `mappend` w'
    w'' `seq` return (b, w'')

所以我的问题是:这是否存在于某些“标准”库中?如果没有,为什么不呢?

1 个答案:

答案 0 :(得分:16)

您问题的直接答案是:不,没有标准库提供此功能。此外,您提出的版本仍将泄露。我知道唯一不泄漏的版本是使用严格WriterT来模拟StateT。我写了一篇非常详细的e-mail about this to the Haskell libraries mailing list来比较几种实现的严格性和性能。简而言之:将WriterT实现为严格StateT不仅可以消除空间泄漏,还可以生成非常高效的代码。

以下是有效的实施方案:

newtype WriterT w m a = WriterT { unWriterT :: w -> m (a, w) }

instance (Monad m, Monoid w) => Monad (WriterT w m) where
    return a = WriterT $ \w -> return (a, w)
    m >>= f  = WriterT $ \w -> do
        (a, w') <- unWriterT m w
        unWriterT (f a) w'

runWriterT :: (Monoid w) => WriterT w m a -> m (a, w)
runWriterT m = unWriterT m mempty

tell :: (Monad m, Monoid w) => w -> WriterT w m ()
tell w = WriterT $ \w' ->
    let wt = w `mappend` w'
     in wt `seq` return ((), w `mappend` w')

我希望在某些时候将此添加到transformers,但是有一些小问题需要解决(例如模块名称应该是什么)。