2015-03-03 62 views
3

作为标题,我有一个ruby程序处理大量的数据。该计划把所有的内存和有系统命令hostname在里面, 的主叫和错误发生如何强制Ruby释放内存到操作系统

Cannot allocate memory - hostname

我试过GC.start,它不工作。

那么我该如何强制ruby释放未使用的内存呢?

好的,这是来自其他人的测试代码,最后的错误表明big_var已被回收。但内存仍未释放。

require "weakref" 
def report 
    puts "#{param}:\t\t Memory " + `ps ax -o pid,rss | grep -E "^[[:space:]]*#{$$}"` 
     .strip.split.map(&:to_i)[1].to_s + 'KB' 
end 
big_var = "" 
#big_var = WeakRef.new(big_var) 

report 
big_var = 1_000_000.times.map(&:to_s) 
report 

big_var = WeakRef.new(big_var) 

GC.start 

sleep 1 
report 

p big_var.length 

#Memory 7508KB 
#Memory 61516KB 
#Memory 53700KB 
#test.rb:20:in `<main>': Invalid Reference - probably recycled (WeakRef::RefError) 

OK,我想事情闹,我不明白为什么GC.stat[:heap_used]仍是大我没有$big_var=nilGC.start

puts GC.stat[:heap_used] 
$big_var = [] 
    5000000.times { |i| 
    $big_var << i.to_s 
    } 
puts GC.stat[:heap_used] 
$big_var = nil 
puts GC.stat[:heap_used] 
GC.start 
puts GC.stat[:heap_used] 

#70 
#12286 
#12286 
#9847  

此外后,我使用Ruby 2.1和CentOS 6.4

+0

http://ruby-doc.org/stdlib-2.1.2/libdoc/weakref/rdoc/WeakRef.html – 2015-03-03 08:04:20

+0

它不起作用。 – sou 2015-03-03 08:25:44

+0

告诉我们代码sou – peter 2015-03-03 08:33:45

回答

2

我可以看到垃圾回收是由WeakRef类完成的。这里是我的尝试证明这一点:

require 'objspace' 
require "weakref" 

big_var = "" 
puts "memory size: #{ObjectSpace.memsize_of big_var}" 

big_var = 1_000_000.times.map(&:to_s) 
puts "memory size: #{ObjectSpace.memsize_of big_var}" 

big_var = WeakRef.new(big_var) 
GC.start 

puts "memory size: #{ObjectSpace.memsize_of big_var}" 

输出

[[email protected]_ruby (master)]$ ruby a.rb 
memory size: 40 
memory size: 11636312 
memory size: 40 
[[email protected]_ruby (master)]$ 

看看这个方法:memsize_of

+0

是的,我可以得到相同的输出,即使只是设置big_var为零。但内存仍然没有发布~~~ – sou 2015-03-03 09:14:39

+0

我认为memsize_of只是big_var现在引用的内存大小。但是之前引用的大内存仍然由Ruby保存,而没有发布。 – sou 2015-03-03 09:18:02

+0

@sou好的可能是..我不确定:/ – 2015-03-03 09:18:44