所以我们有:
import Control.Monad.Writer.Strict
type M a = Writer (Map Key Val) a
某些Key
和Val
的。
只要我们不查看收集的输出,一切正常:
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'')
所以我的问题是:这是否存在于某些“标准”库中?如果没有,为什么不呢?
答案 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
,但是有一些小问题需要解决(例如模块名称应该是什么)。