Космические утечки и Писатели, и Суммы (о мой!) - PullRequest
25 голосов
/ 11 октября 2011

Недавно я играл с Writer Monad и столкнулся с тем, что кажется космической утечкой.Пока не могу сказать, что полностью понимаю эти вещи, поэтому мне хотелось бы знать, что здесь происходит и как это исправить.

Во-первых, вот как я могу вызвать эту ошибку:

import Control.Monad.Writer
import Data.Monoid

foo :: Integer -> Writer (Sum Integer) Integer
foo 0 = return 0
foo x = tell (Sum x) >> foo (pred x)

main = print $ runWriter $ foo 1000000

Я получаю:

Stack space overflow: current size 8388608 bytes.
Use `+RTS -Ksize -RTS' to increase it.

Чтобы лучше это понять, я переопределил аналогичную функциональность без Writer или Sum, и если я сделаю все красиво и лениво, я получу ту же ошибку:

bar :: Integer -> (Integer, Integer)
bar x = bar' 0 x
    where bar' c 0 = (0, c)
          bar' c x = bar' (c + x) (pred x)

Но я могу исправить это, добавив seq к уравнению:

bar' c x = c `seq` bar' (c + x) (pred x)

Я пытался seq использовать различные биты моей foo функции, но это не таккажется, чтобы помочь.Кроме того, я пытался использовать Control.Monad.Writer.Strict, но это тоже ничего не меняет.

Нужно ли как-то быть строгим Sum?Или я упускаю что-то совершенно другое?

Примечания

  • Возможно, моя терминология здесь неверна.Согласно зоопарк утечки пространства , моя проблема была бы классифицирована как «переполнение стека», и если бы это было так, как бы я преобразовал foo в более итеративный стиль?Является ли моя ручная рекурсия проблемой?
  • После прочтения Переполнение пространства Haskell у меня была идея скомпилировать с -O2, просто чтобы посмотреть, что произойдет.Это может быть темой для другого вопроса, но с оптимизацией даже моя функция seq 'd bar не запускается. Обновление : эта проблема исчезнет, ​​если я добавлю -fno-full-laziness.

Ответы [ 3 ]

12 голосов
/ 12 октября 2011

Проблема с монадой Writer в том, что она >>= не является хвостовой рекурсивной:

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

Как видите, для оценки mappend необходимо оценить как m, так и k a, что означает, что весь стек рекурсивных вызовов должен быть принудительно обработан, прежде чем можно будет оценить первую mappend. Я считаю, что независимо от строгости, монада Writer вызовет переполнение стека в вашем определении (или его можно избежать с помощью lazy version как-нибудь?).

Если вы все еще хотите использовать монады, вы можете попробовать State, который является хвост-рекурсивным. Либо строгая версия этого со строгим put:

import Control.Monad.State.Strict

foo :: Integer -> State (Sum Integer) Integer
foo 0 = return 0
foo x = do
   w <- get
   put $! w `mappend` (Sum x)
   foo (pred x)

main = print $ (`execState` Sum 0) $ foo 1000000

Или ленивая версия со стилем передачи продолжения (CPS):

import Control.Monad.Cont
import Control.Monad.State.Lazy

foo :: Integer -> ContT Integer (State (Sum Integer)) Integer
foo 0 = return 0
foo x = do
  w <- get
  put $! w `mappend` (Sum x)
  foo (pred x)

main = print $ ((`execState`Sum 0) . (`runContT`return)) $ foo 1000000

Удобный аналог для tell:

stell :: (MonadState w m, Monoid w) => w -> m ()
stell a = get >>= \w -> put $! w `mappend` a

Я подозреваю, что если бы можно было использовать ContT с Writer, то CPS также помог бы нам с Writer, но похоже, что определить ContT для MonadWriter невозможно

7 голосов
/ 11 октября 2011

Посмотрите на источник для монады строгого писателя: http://hackage.haskell.org/packages/archive/transformers/0.2.2.0/doc/html/src/Control-Monad-Trans-Writer-Strict.html#line-122

Отличие от ленивого писателя состоит в том, что в последнем случае сопоставление с образцом лениво - но ни в одном случае операция mappend или "состояние" писателя пока не являются принудительными. Чтобы решить вашу проблему, вам понадобится «очень строгий» писатель.

4 голосов
/ 11 октября 2011

Я думаю, что вы понимаете правильно.

Мне интересно, как эти функции:

bar :: Integer -> (Integer, Integer)
bar x = bar' 0 x
    where bar' c 0 = (0, c)
          bar' c x = c `seq` bar' (c + x) (pred x)
      --  bar' c x = let c' = c+x in c' `seq` bar' c' (pred x)
      --  bar' !c !x = bar' (c+x) (pred x)

производит переполнение стека при компиляции с оптимизацией, хотя связанные функции:

bar2 :: Integer -> (Integer, Integer)
bar2 x = bar' 0 x
     where bar' c 0 = (0, c)
           bar' !c !x = let c' = c+x in c' `seq` bar' c' (pred x)

bar3 :: Integer -> Integer
bar3 x = bar' 0 x
     where bar' c 0 = c
           bar' c x = c `seq` bar' (c + x) (pred x)

bar4 :: Integer -> (Integer, Integer)
bar4 x = (0, bar' 0 x)
     where bar' c 0 = c
           bar' c x = c `seq` bar' (c + x) (pred x)

нет.

Я думаю, что это похоже на ошибку в оптимизаторе GHC, и вы должны сообщить об этом . Если посмотреть на ядро ​​(созданное с помощью -ddump-simpl), аргумент c не всегда строго оценивается для переполняющих функций. bar2 - самая близкая рабочая версия, которую я нашел к оригиналу, и она кажется мне чрезмерно определенной.

Поскольку у вас очень простой тестовый пример, есть большая вероятность, что он будет исправлен.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...