- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我看到很多 Forth 代码只是在做 s"Hello "s"world"s+
好像没什么,但现在我想了一下,这实际上分配了 3 个指针,并丢失了两个指针他们走向伟大的虚无。 slurp-file
的大多数使用都存在同样的问题。
但是,如果我需要将分配的每个字符串地址放入临时位置以便稍后释放它们,例如 s"foo"over >r ( ... do some ... ) r> free
,我要失去理智了。这方面的最佳实践是什么?
我没有看到很多 Forth 代码考虑内存分配,而且它的堆栈方面似乎处于一种“即发即忘”的情绪。
我正在开发一个提供 HTML 文件的 Web 服务器,虽然请求保存在可重用的 pad
中,但另一方面,响应是混合文件和字符串连接。
这意味着,如果我让服务器在互联网上运行一段时间,并让你在那里找到的各种机器人玩它,我可能会丢失大量的内存,只是为了回答它们离开。
因此,我向这里活跃的 Forth 社区寻求最佳实践。
我应该:
(1) 是我最可怕的噩梦中的一个场景
(2)是一种偷懒的方式,但也不是那么糟糕
(3) 我查看了代码,这对我来说似乎太过分了
(4) 是我真正想要的目标,但有点雄心勃勃
here
相同的词来指向其中的下一个空闲位置。s+
和其他字符串操作字编写新版本,将其添加到此处
,并按其大小递增这是一个好的策略吗?我错过了什么吗?
最佳答案
简短的回答:你保留一个列表。
s"
并不总是分配新内存我对 s"
工作原理的解释是错误的。在解释时(在读取文件期间或在交互式终端中),它有效地分配
内存,因此你在堆栈上得到一个字符串。但是当 s"
被编译成一个单词时,它的执行会调用 allot
,这会使用现有的字典空间。
Gforth 0.7.3
see s"
34 parse save-mem ; \ interpret
34 parse POSTPONE SLiteral ; \ compile
see save-mem
swap >r dup allocate throw swap 2dup r> -rot move ;
see SLiteral
tuck 2>r POSTPONE AHEAD here 2r> mem, align >r POSTPONE THEN r> POSTPONE Literal POSTPONE Literal ;
see mem,
here over allot swap move ;
POSTPONE AHEAD
允许在读取字符串期间只调用一次进行分配的代码,然后在执行过程中跳过这一部分,直接进入将地址和长度压入字符串的部分堆栈。
这意味着代码中内联的字符串会就地编译,不需要释放。
s" This string should be freed" \ on the heap
: hello ( -- addr u )
s" This one should not." ; \ in dictionary space
s"
是实现定义的有些fors为所有s"
调用重用相同的缓冲区,而其他一些fors则允许您同时访问2或3个字符串,但下一个将删除现有数据。
因此,您不应将 s"
字符串视为理所当然,如果您想保留它,则应复制它。
因此,主要问题不是 s"
的使用,而主要是 s+
和 slurp-file
,它们都调用 内部分配
。
我使用所谓的“空闲列表”解决了这个问题。每次我使用 s+ 或 slurp-file 时,我都会保留对给定指针的引用,并将其存储在链接列表中以便稍后释放。
\ a simple linked-list keeping track of allocated strings
variable STRBUF-POINTER \ the current head of the list
0 STRBUF-POINTER !
struct
cell% field strbuf-prev \ previous entry
cell% field strbuf-addr \ the string allocated
end-struct strbuf%
: add-strbuf ( addr -- )
strbuf% %alloc >r
( addr ) r@ strbuf-addr !
STRBUF-POINTER @ r@ strbuf-prev !
r> STRBUF-POINTER ! ; \ become the new head
: (?free) ( addr -- )
dup if free throw else drop then ;
: free-strbuf ( -- ) \ walk up the list and free strings
begin
STRBUF-POINTER @
while
STRBUF-POINTER @ >r
r@ strbuf-addr @ (?free) \ free the string
r@ strbuf-prev @ STRBUF-POINTER ! \ prev becomes new head
r> (?free) \ free the struct itself
repeat ;
: my-s+ ( $1 $2 -- $3 )
s+ over add-strbuf ;
: my-slurp-file ( $path -- $content )
slurp-file over add-strbuf ;
: main-process
begin
listen \ wait for client request
( ... use my-s+ and my-slurp-file ... )
send-response
free-strbuf \ we free everything we used
again
;
在我的例子中,这个解决方案似乎足以大幅减少内存使用量。但在某些情况下,您可能希望通过实现regions来改进它:不要在链接列表中为每个字符串创建一个新元素,而是让它们跟踪大的可重用缓冲区,就像我在解决方案(4)中所说的那样。
关于memory-management - 如何跟踪 Forth 中分配的所有字符串并按时释放它们?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/72819086/
在 ARM 中,内存类型指定为: 正常 设备 强烈有序 在Device type里面,好像这个类型也可以区分 不可共享的设备内存 可共享设备内存 不可共享和可共享设备内存有什么区别?我们如何分别使用这
在 ARM 中,内存类型指定为: 正常 设备 强烈有序 在Device type里面,好像这个类型也可以区分 不可共享的设备内存 可共享设备内存 不可共享和可共享设备内存有什么区别?我们如何分别使用这
This diagram很清楚不同YARN和Spark内存相关设置之间的关系,除了spark.python.worker.memory。 spark.python.worker.memory 如何适应
我正在尝试使用复杂的if-else决策树来实现GLSL片段着色器。不幸的是,着色器编译器很早就失败,并出现“语法错误-内存耗尽”错误。 GLSL中的代码大小或决策树深度是否有任何限制?有什么建议如何克
什么是“标记内存”,它如何帮助减小程序大小? 最佳答案 您可能指的是 tagged union ,或更具体地说是硬件实现,如 LISP 机器中使用的标记架构。基本上是一种存储具有类型信息的数据的方法。
我的内存有问题。我不明白为什么当我的程序长时间运行时 Go 使用越来越多的内存(从不释放它)。 第一次分配后,程序使用了将近 9 MB 的内存。然后在 12 小时后,它开始以指数方式使用更多内存,直到
在 Windows 机器上,MATLAB 用户可以使用 memory或 feature memstats命令。但是,这些都不能在机器上工作,失败如下: >> memory??? Error using
引导 Linux 内核时,可以在 RAM 中加载 initramfs 存档和 DTB 文件,并将这些物理地址指定给内核。例如,使用 U-Boot,您可以执行以下操作: bootz 0x80008000
我正在学习虚拟内存的概念,但是这个问题让我困惑了一段时间。由于大多数现代计算机都使用虚拟内存,因此当程序正在执行时,操作系统应该在 RAM 和磁盘之间将数据分页进出。但为什么我们仍然遇到“内存不足”的
我在 Colab Pro+(使用高 RAM 选项)上运行神经网络时发现了这个问题。 运行时错误:CUDA 内存不足。尝试分配 8.00 GiB(GPU 0;15.90 GiB 总容量;12.04 Gi
当我在任何地方阅读基于操作系统的书籍时,考虑到时间限制和开销很高,从内存和 I\O(子系统)获取数据是昂贵的,这就是为什么在某些硬件制造商中提供一些其他方式来访问它们,如ARM7 some ISAs像
据我所知,ADS v.10 尝试将查询结果保留在内存中,直到它变得非常大。对于 __output 表和临时表也应该如此。当结果变大时,交换声明。 问题是为查询、 worker 等设置了什么内存限制?可
序言 我正在写一个小演示文稿来列出使用 Docker 时的一些“陷阱”,我也遇到了自己的一个问题。 在解释让 Docker 在没有内存限制的情况下运行的危险时,我发现它的行为不像我预期的那样。 我使用
我们有一个 ASP.NET 项目(40 个左右的 Web 表单、50 个表、相当标准的 IO 内容,并尽可能减少),很快需要部署。系统上大约有 100 个并发用户,但任何时候只有大约 20 个用户在使
我在 dotcloud 上使用 redis 内存存储,但尽管 key 已过期,但它的 used_memory 再也不会下降。从 redis-cli 使用 flushdb 或 flushall 不会导致
我使用的是 Xcode 10.2.1 和 macOS Catalina Developer Beta 2。每当我尝试使用内存图调试器时,我都会收到此错误: Memory Graph Debugger:
所以我一直在寻找这个问题的解决方案有一段时间了。我编写了一个程序来从两个单独的文本文件中获取数据,对其进行解析,然后输出到另一个文本文件和一个 ARFF 文件以供 Weka 分析。我遇到的问题是我编写
对不起,我对 erlang 文档中的以下描述不太清楚: erlang:memory() -> [{Type, Size}] with Type: "total" means: "The total a
在查看示例合约时,有时会在带有“内存”的方法中声明数组,有时则不会。有什么区别? 最佳答案 如果没有内存关键字,Solidity会尝试在存储中声明变量。 首席 Solidity 开发者 chriset
我不明白Matlab并行计算工具箱中的parfor cicle是如何与内存一起工作的:我读到它在所有worker之间共享内存(然后我认为每个worker(核心)都可以访问感兴趣的内存位置而无需制作本地
我是一名优秀的程序员,十分优秀!