如何通过反射获取Ruby的Module类定义的常量?

时间:2010-02-22 06:46:19

标签: ruby reflection metaprogramming

我试图让Matz和Flanagan的“Ruby Programming Language”元编程章节进入我的脑海,但是我无法理解我梦寐以求的以下代码片段的输出:

p Module.constants.length           # => 88
$snapshot1 = Module.constants       
class A
  NAME=:abc

  $snapshot2 = Module.constants
  p $snapshot2.length               # => 90
  p $snapshot2 - $snapshot1         # => ["A", "NAME"]

end
p Module.constants.length           # => 89
p Module.constants - $snapshot1     # => ["A"]
p A.constants                       # => ["NAME"]

本书指出类方法constants返回类的常量列表(正如您在A.constants的输出中看到的那样)。 当我遇到上述奇怪的行为时,我试图获取为Module类定义的常量列表。

A的常量显示在Module.constants中。如何获取Module类定义的常量列表?

docs

  

Module.constants返回系统中定义的所有常量。包括所有类和方法的名称

由于AModule.constants继承了它的实现,它在基类和派生类型中的表现如何?

p A.class               # => Class
p A.class.ancestors       # => [Class, Module, Object, Kernel]

注意:如果您使用的是Ruby 1.9,constants将返回一个符号数组而不是字符串。

2 个答案:

答案 0 :(得分:52)

好问题!

您的困惑是因为类方法Module.constants隐藏了Module#constants的实例方法Module

在Ruby 1.9中,通过添加可选参数解决了这个问题:

# No argument: same class method as in 1.8:
Module.constants         # ==> All constants
# One argument: uses the instance method:
Module.constants(true)   # ==> Constants of Module (and included modules)
Module.constants(false)  # ==> Constants of Module (only).

在上面的示例中,A.constants调用了Module#constants(实例方法),Module.constants调用了Module.constants

在Ruby 1.9中,您希望调用Module.constants(true)

在Ruby 1.8中,可以在#constants上调用实例方法Module。您需要获取实例方法并将其绑定为类方法(使用不同的名称):

class << Module
  define_method :constants_of_module, Module.instance_method(:constants)
end

# Now use this new class method:
class Module
   COOL = 42
end
Module.constants.include?("COOL")  # ==> false, as you mention
Module.constants_of_module         # ==> ["COOL"], the result you want

我希望我能够将我的backports gem的1.9功能完全反向移植到1.8,但我想不出在Ruby中只获取模块的常量(不包括继承的模块)的方法1.8。

编辑:刚刚更改了官方文档,以正确反映这一点......

答案 1 :(得分:3)

在Marc的回应之后,我不得不回到我的思考洞穴一段时间。使用更多代码片段进行修补,然后再进行更多操作。最后,当Ruby的方法分辨率似乎有意义时,将其写成博客文章,以便我不会忘记。

符号:如果 A“ A 的本征类

调用A.constants时,方法解析(请参阅my blog post中的图像以获得视觉辅助)按顺序查找以下位置

  • MyClass"Object"BasicObject"(单身方法)
  • Class(实例方法)
  • Module(实例方法)
  • Object(实例方法)和内核
  • BasicObject(实例方法)

Ruby找到实例方法Module#constants

调用Module.constants时,Ruby会查看

  • Module"Object"BasicObject"(单身方法)
  • Class(实例方法)
  • Module(实例方法)
  • Object(实例方法)和内核
  • BasicObject(实例方法)

这次,正如Marc所说,Ruby在Module".constants找到了单例/类方法。

模块定义了一个影响实例方法的单例方法。 singleton方法返回所有已知常量,而instance方法返回当前类及其祖先中定义的常量。