- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
简介
以下代码显示,当使用runhaskell
时,Haskell垃圾收集器会在a
不再使用时释放内存。它会在释放变量 a
时导致核心转储 - 出于某种目的,检查行为 - a
将 nullFunPtr
作为终结器。
module Main where
import Foreign.Ptr
import Foreign.ForeignPtr
main :: IO ()
main = do
a <- newForeignPtr nullFunPtr nullPtr
putStrLn "Hello World"
问题
在 ghci 中运行相同的内容时,它不会释放内存。如何强制 ghci 释放不再使用的变量?
$ ghci
> import Foreign.Ptr
> import Foreign.ForeignPtr
> import System.Mem
> a <- newForeignPtr nullFunPtr nullPtr
> a <- return () -- rebinding variable a to show gc that I'm no longer using it
> performGC
> -- did not crash - GC didn't release memory
> ^D
Leaving GHCi.
[1] 4396 segmentation fault (core dumped) ghci
退出时内存被释放,但这对我来说已经太晚了。我正在扩展 GHCi 并将其用于其他目的,我需要尽早释放内存 - 按需或尽可能快地释放内存真的很棒。
我知道我可以调用 finalizeForeignPtr
,但我使用 foreignPtr
只是为了调试目的。在最后一个例子中,我如何才能释放a
?
如果无法使用 ghci 提示符执行此操作,我还可以修改 ghci
代码。也许我可以通过 modyfing ghci Interactive Context 发布这个 a
或DynFlags ?到目前为止,我的研究还没有进展。
最佳答案
追踪代码我们发现该值存储在PersistentLinkerState
数据类型的closure_env
字段中,它是一个ClosureEnv
,即从名称到 HValue 的映射。 Linker.hs
中的相关函数是
extendLinkEnv :: [(Name,HValue)] -> IO ()
-- Automatically discards shadowed bindings
extendLinkEnv new_bindings =
modifyPLS_ $ \pls ->
let new_closure_env = extendClosureEnv (closure_env pls) new_bindings
in return pls{ closure_env = new_closure_env }
虽然注释表明它应该删除阴影绑定(bind),但它没有,至少没有按照您希望的方式删除。
原因是,正如AndrewC正确编写的:虽然两个变量具有相同的源代码名称,但它们对于编译器来说是不同的(它们附加了不同的Unique
)。在向上面的函数添加一些跟踪后,我们可以观察到这一点:
*GHCiGC> a <- newForeignPtr nullFunPtr nullPtr
extendLinkEnv [a_azp]
*GHCiGC> a <- return ()
extendLinkEnv [a_aF0]
*GHCiGC> performGC
extendLinkEnv [it_aFL]
此时删除具有相同源名称的绑定(bind)应该可以解决您的 GC 问题,但我不太了解编译器,无法判断还会破坏什么。我建议你开个票,希望有人知道。
绑定(bind)与值(value)的混淆
在评论中似乎对绑定(bind)和值有些困惑。考虑这段代码:
> a <- return something
> b <- return somethingelse
> a <- return (b+b)
> b <- return anewthing
在当前的实现中,堆将由`
组成某事
其他东西
(+)
运算符和 somethingelse
新事物
。此外,解释器的环境引用了所有四个堆值,因此无法对任何内容进行 GC。
remdezx 正确地期望 GHCi 会删除对 something
和 somethingelse
的引用。反过来,这将允许运行时系统垃圾收集某些东西
(我们假设没有进一步的引用)。 GHCi 仍然引用 thunk,而 thunk 又引用 somethingelse
,因此这不会被垃圾收集。
显然,这个问题是非常具体的实现,这个答案也是如此:-)
关于haskell - 如何让 GHCI 释放内存,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26712188/
我经常使用 ghci 进行少量计算,并使用 stack ghci 来处理我的实际项目。 为了使第一个更容易,我编写了一个 .ghci 文件,其中包含许多导入的模块,但其中一些模块不存在于我的堆栈项目中
我试图用这个来完善我的 GHCi:http://www.reddit.com/r/haskell/comments/144biy/pretty_output_in_ghci_howto_in_comm
我有一个 .ghci在我的本地项目目录中,另一个在我的 $HOME 中.当我做 stack ghci ,然后 $HOME/.ghci首先加载,然后是 $PWD/.ghci .是否可以只加载本地 .gh
在尝试将 ~/.ghci 文件更改为我的配置时,当我打开 GHCi 时出现此错误。 GHCi, version 7.6.3: http://www.haskell.org/ghc/ :? for h
所以我正在完成 Real World Haskell 的一些初始章节练习,我想知道 GHCi 中是否有一个选项可以让它在每个递归调用上显示带有参数的函数评估。例如,我写了一个简单版本的“map”,当我
我是 Haskell 的新手,在调试时我遇到了一个烦人的行为。 我使用 :break 添加断点 我运行 main 一切正常 我输入:继续完成执行 当我重新运行 main 时,断点不再命中,但断点没有被
从 ghc 7.6 更新到 7.10 后,您似乎无法 :m [Module]或 ghci> import [Module]其中 [Module.hs] 是您的手写模块文件,位于当前工作目录中。 似乎
我正在尝试编写一个 Erasthosthenes 函数的筛选器,该函数为用户提供从 2 到其上限的所有质数。所以我写了这段代码: main = do putStrLn "Upper Limit" g
这个问题在这里已经有了答案: How does GHCi pick names for type variables? (1 个回答) How are variable names chosen in
我现在应该真的知道这一点,但我不知道。我经常开发基于 Cabal 的软件包,并且刚刚成功运行了 cabal build .现在我想在 GHCi 中尝试一些东西。如果我运行 cabal repl ,然后
编写一个模块: module Foo where foo = 3.14 编译它: ghc -c Foo.hs 加载它: ghci -ignore-dot-ghci GHCi, version 7.8.
在检查不同整数类型的大小( minBound 、 maxBound )和“十进制表示的长度”时,我碰巧看到了一些奇怪的行为。 使用 GHCi: Prelude> :{ Prelude| let mi
考虑程序: l = [0..10] l' = map (+1) [0..10] 使用 GHCi 运行它,并键入 :sprint l 和 :sprint l' 将显示两个列表都未计算。但是,在运行 le
GHCi 中有没有办法基本上获得状态转储?我的意思是一个列表: 所有加载的运算符及其优先级、关联性和签名。 所有加载的类。 所有加载的数据、类型和新类型以及它们是哪些类的实例。 所有加载的函数都带有它
我刚开始学习 Haskell,很难理解 Haskell 程序的“流程”。 例如在 Python 中,我可以编写一个脚本,将其加载到解释器并查看结果: def cube(x): return x
A做了一个模块Timeit。我无法将其导入 GHCi。 模块: module Timeit (timeit, timeCatch) where import Data.Time.Clock timei
我刚刚安装了 Haskell Platform for Windows(版本 2011.2.0.1),并开始通过 HaskellQuestions.pdf 工作 第二个问题需要“x = 3”作为答案。
作为 Haskell 的新手,我正在努力解决以下差异(我确信这是有充分理由的)。也许我的问题只是源于对 GHCi 的误解,但当我能消除疑虑时,我晚上会睡得更好。 来了。如果,在将名称 foo 绑定(b
我正在试验 GHCi 的 :sprint命令。考虑以下: GHCi> xs = [1..10] :: [Int] GHCi> :sprint xs xs = _ GHCi> length xs 10
假设我有以下功能: minc = map (+1) natural = 1:minc natural 它似乎是这样展开的: 1:minc(1:minc(1:minc(1:minc(1:minc(1:m
我是一名优秀的程序员,十分优秀!