【问题标题】:Object memory allocation对象内存分配
【发布时间】:2014-03-17 16:16:25
【问题描述】:

我想知道 Ruby 中的内存分配是如何工作的。

在 Ruby 中,我们可以打开一个类定义并添加更多实例变量/方法。在这种情况下,新对象将需要比现有实例更多的内存。它如何在内部跟踪分配给实例的内存?

例如,我们有以下类定义:

class MyClass
    def myMethod
        @a = 5
    end
end

我们创建了MyClass的实例:

m = MyClass.new
m.myMethod

此时,m 将被分配一些内存。现在,我们打开类定义并添加另一个实例变量b

class MyClass
    def mySecondMethod
        @b = 5
    end
end

这个新方法mySecondMethod和实例变量bMyclass的现有实例m中也可用。

调用mySecondMethod实例化并初始化b

m.mySecondMethod

Ruby 如何在内部管理内存?它是重新分配内存块,例如m,还是维护某种指向新分配内存的指针?

【问题讨论】:

  • 只有一点 - mySecondMethodb 这样的实例上不可用,而是在 b 这样的实例类中可用。
  • @ArupRakshit,我不明白你的意思,“像b 这样的实例”。你能改写一下吗?谢谢。

标签: ruby object memory-management


【解决方案1】:

答案取决于实际的实现。这里我假设你问的是MRI

Ruby 对象分配在heap 上。在谈论对象分配时,没有stack 的概念。

堆被分成,每页由16kb组成。每个页面都被分割成 固定大小 插槽,这些插槽可以容纳 Ruby 对象。一个页面可以容纳大约 408 个对象,因为每个对象(这是一个 RVALUE 结构)占用 40 个字节。

所有这些都由 VM 管理(即YARV)。

来源:http://timetobleed.com/garbage-collection-slides-from-la-ruby-conference/

关于您的示例,变量只是保存对对象的引用,因此m 实际上指向已分配的MyClass 对象。

在内部备份 MyClass 的 C 结构 (RClass) 包含指向具有用户定义方法(如 #mySecondMethod)的表的指针和指向具有其实例变量名称的表的指针对象有。

每个对象(由RObject 备份,因为Object 类是所有对象的默认根)内部包含一个pointer to the values of its instance variables

新定义的#mySecondMethod 是可用的,因为该语言的动态特性以及方法查找发生在运行时这一事实。

【讨论】:

  • @Agis 你的回答很好。我有一个关于“所有这些都由 VM(即 YARV)管理”的查询。这个说法。 YARV 文档(en.wikipedia.org/wiki/YARV) 显示这是作为 Ruby 1.9 版本的一部分添加的。那么在之前版本的ruby中,ruby对象是如何管理堆内存的呢?
【解决方案2】:

你的第二个猜测是正确的。 Ruby 对象维护指向堆内存的指针。

但是请注意,在您的示例中,因为从未在您的对象 m 上调用 mySecondMethod,所以不会为 m 设置/初始化该实例变量 @b

可能对您最有帮助的事情是,首先,多练习使用 Ruby 并了解它的类和对象习语是如何工作的。然后,我建议查看一些关于 Ruby 垃圾收集器的资源,例如 this very good and recent article

【讨论】:

  • 感谢您的链接。我会通过它。实际上,忘记在原始问题中调用 mySecondMethod 。编辑问题以拨打电话。
猜你喜欢
  • 2015-10-15
  • 2013-01-21
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2013-12-04
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多