我试图让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
返回系统中定义的所有常量。包括所有类和方法的名称
由于A
从Module.constants
继承了它的实现,它在基类和派生类型中的表现如何?
p A.class # => Class
p A.class.ancestors # => [Class, Module, Object, Kernel]
注意:如果您使用的是Ruby 1.9,constants
将返回一个符号数组而不是字符串。
答案 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)
符号:如果 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方法返回当前类及其祖先中定义的常量。