这样做的原因是包含的方法块实际上是在类的上下文中评估的。这意味着,其中定义的该方法是在包含模块时在类上定义的,因此优先于包含的模块。
module Child1
extend ActiveSupport::Concern
included do
def foo
end
end
end
module Child2
def bar
end
end
class A
include Child1
include Child2
end
A.new.method(:foo).owner #=> A
A.new.method(:bar).owner #=> Child2
方法查找
在ruby中,每次要调用一个方法,ruby都要先找到它(不知道是方法还是变量)。它是通过所谓的方法查找完成的。当没有指定接收者(纯调用 like puts
)时,它首先在当前范围内搜索任何变量。找不到时,它会在 current 上搜索该方法self
。当指定接收器时(foo.bar
),它自然会搜索给定接收器上的方法。
现在查找 - 在 ruby 中,所有方法总是属于某个模块/类。顺序中的第一个是接收者的特征类(如果存在)。如果不是,则常规接收者的班级是第一位的。
如果在类中找不到该方法,则它会以相反的顺序搜索给定类中所有包含的模块。如果在那里没有找到任何东西,那么给定类的超类是下一个。整个过程递归进行,直到找到东西。当查找到达 BasicObject 并且找不到方法时,它会退出并触发对 method_missing 的搜索,默认实现是在 BasicObject 上定义的。
需要注意的重要一点是,属于类的方法总是优先于模块方法:
module M
def foo
:m_foo
end
end
class MyClass
def foo
:class_foo
end
include M
end
MyClass.new.foo #=> :class_foo
关于super
搜索超级方法非常相似 - 它只是尝试查找具有相同名称的方法,该方法在方法查找中更进一步:
module M1
def foo
"M1-" + super
end
end
module M2
def foo
'M2-' + super
end
end
module M3
def foo
'M3-' + super
end
end
class Object
def foo
'Object'
end
end
class A
include M2
include M3
end
class B < A
def foo
'B-' + super
end
include M1
end
B.new.foo #=> 'B-M1-M3-M2-Object'
ActiveSupport::Concern#included
included
是一个非常简单的方法,它接受一个块并self.included
在当前模块上创建一个方法。该块使用 执行instance_eval
,这意味着其中的任何代码实际上都是在包含给定模块的类的上下文中执行的。因此,当您在其中定义一个方法时,该方法将由包含该模块的类拥有,而不是模块本身。
每个模块只能包含一个具有给定名称的方法,一旦您尝试定义第二个具有相同名称的方法,之前的定义将被完全删除,并且无法使用 ruby 方法查找找到它。由于在您的示例中,您在包含块中包含了两个具有相同方法定义的模块,因此第二个定义完全覆盖了第一个,并且在方法查找中没有其他定义更高,因此 super 肯定会失败。