维基百科写到 Hylomorphism :
(我加粗标注)
使用 recursion-schemes 库
我写了一个非常简单的hylomorphism:
import Data.Functor.Foldable
main :: IO ()
main = putStrLn $ show $ hylosum 1000
hylosum :: Int -> Int
hylosum end = hylo alg coalg 1
where
-- Create list of Int's from 1 to n
coalg :: Int -> ListF Int Int
coalg n
| n > end = Nil
| otherwise = Cons n (n + 1)
-- Sum up a list of Int's
alg :: ListF Int Int -> Int
alg Nil = 0
alg (Cons a x) = a + x
在 cabal 文件中,我指示 GHC 优化代码:
name: Hylo
version: 0.1.0.0
synopsis: Hylomorphisms and Deforestation
build-type: Simple
cabal-version: >=1.10
executable Hylo
main-is: Main.hs
ghc-options: -O2
build-depends: base >=4.10 && <4.11 , recursion-schemes
default-language: Haskell2010
使用 stackage lts-10.0 (GHC 8.2.2) 我用
stack build
编译并用 stack exec Hylo -- +RTS -s
运行,我得到:500500
84,016 bytes allocated in the heap
3,408 bytes copied during GC
44,504 bytes maximum residency (1 sample(s))
25,128 bytes maximum slop
2 MB total memory in use (0 MB lost due to fragmentation)
现在我将
hylosum 1000
更改为 hylosum 1000000
(1000 倍),我得到:500000500000
16,664,864 bytes allocated in the heap
16,928 bytes copied during GC
15,756,232 bytes maximum residency (4 sample(s))
29,224 bytes maximum slop
18 MB total memory in use (0 MB lost due to fragmentation)
因此堆使用量从 84 KB 增加到 16,664 KB。这是以前的 200 倍。
因此我认为,GHC 不会做维基百科中提到的森林砍伐/融合!
这并不奇怪:变形从左到右创建列表项
(从 1 到 n)并且 catamorphism 从右到左以相反的方式消耗项目
(从 n 到 1)并且很难看出 hylomorphism 是如何工作的
无需创建整个中间列表。
问题:
GHC 是否能够执行森林砍伐?
如果是,我必须在我的代码或 cabal 文件中更改什么?
如果是,它是如何真正起作用的?
如果不是,问题出在哪里:在维基百科、GHC 还是在图书馆?
最佳答案
数据结构实际上被融合了,但生成的程序不是尾递归的。优化后的代码基本上是这样的(看不到 Cons
或 Nil
):
h n | n > end = 0
| otherwise = n + h (n+1)
要评估结果,您必须首先递归评估
h (n+1)
,然后将结果添加到 n
。在递归调用期间,值 n
必须保持存储在某处,因此我们观察到随着 end
增加内存使用量增加。通过将递归调用置于尾部位置并携带恒定大小的累加器,可以获得更紧密的循环。我们希望代码对此进行优化:
-- with BangPatterns
h n !acc | n > end = acc
| otherwise = h (n+1) (n + acc)
在
hylosum
中,对 (+)
的调用发生在 alg
中,我们将其替换为对将由 hylo
构建的延续的调用。alg :: ListF Int (Int -> Int) -> Int -> Int
alg Nil acc = acc
alg (Cons n go) !acc = go (n + acc)
有了这个,我看到在堆中分配了一个恒定的 51kB。
关于haskell - Hylomorphism 中的森林砍伐,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/49135351/