gpt4 book ai didi

ruby - 如何让模块 mixins 为静态方法工作?

转载 作者:数据小太阳 更新时间:2023-10-29 07:08:55 26 4
gpt4 key购买 nike

假设我有两个模块。是否可以将一个模块包含在另一个模块中,使其表现得像一个混入?

例如:

module A
def self.foo
puts "foo"
bar
end
end

module B
include A
def self.bar
puts "bar"
end
end

B.bar
B.foo

编辑:我意识到我最初把代码抄错了。这些方法需要是静态的。更正后的代码在上面(但不起作用)。

最佳答案

如您所知,它不起作用,但为什么它不起作用是关于 Ruby 对象模型的非常好的一课。

当你创建一个对象的实例时,你创建的是一个新对象,它有一组实例变量和一个指向对象类的指针(以及一些其他的东西,比如一个对象 ID 和一个指向父类(super class)的指针) 但方法本身不在对象的实例中。类定义包含方法列表及其代码(以及指向其自身类的指针、指向其父类(super class)的指针和对象 ID)。

当您调用一个实例的方法时,Ruby 会查找该实例的类,并在该类的方法列表中查找您调用的方法。如果它没有找到它,那么它会在类的父类(super class)中查找。如果在那里找不到它,它会在该类的父类(super class)中查找,直到用完所有父类(super class)。然后它回到第一个类并寻找 method_missing 方法。如果找不到,它会转到父类(super class),依此类推,直到到达它设计用于引发错误的根对象。

例如,假设您有一个 Person 类,您使用变量 bubba 创建该类的实例,如下所示:

class Person
attr_accessor :dob, :name
def age
years = Time.now.year - @dob.year
puts "You are #{years} year#{"s" if years != 1} old"
end
def feed
puts "nom, nom, nom"
end
end
bubba = Person.new
bubba.name = "Bubba"
bubba.dob = Time.new(1983,9,26)

类图看起来像这样: alt text

那么当您创建静态方法、类/模块方法时会发生什么?好吧,请记住,在 Ruby 中几乎所有东西都是对象,模块定义是类 Class 的实例。是的,您输入的代码实际上也是一个实例,它是实时代码。当您使用 def self.method_name 创建类方法时,您是在作为类/模块定义的对象实例中创建方法。

太好了,那么您问的那个类方法是在哪里定义的呢?它是在一个匿名类(又名单例、本征、幽灵类)中定义的,正是出于这个原因而创建的。

回到我们的 Person 类,如果我们像这样在实例 bubba 上添加一个类方法会怎样:

def bubba.drive_pickup
puts "Yee-haw!"
end

该方法被放入专为该实例创建的特殊单例类中,单例的父类(super class)现在是 Person 类。这使得我们的方法调用链看起来像这样: alt text

在实例对象 bubba 上定义的任何其他方法也将放入该单例类中。每个实例对象永远不会超过一个单例类。

所以,总而言之,它不起作用的原因是模块中的静态方法是在单例类中为模块定义的实例定义的。当您包含或扩展模块时,您将添加一个指向模块方法表的指针,而不是模块的单例类实例对象的方法表。

这样想:如果你创建了一个 Z 类型的实例 x 和一个 Z 类型的实例 y,x 应该知道 y 吗?不,除非特别告知,否则不会。因此,混合在另一个模块中的模块也不应该知道恰好将第一个模块作为其父类(super class)的其他对象。

要更好地解释 Ruby 对象模型,请观看由非常博学的人提供的精彩免费视频 Dave Thomas (不,不是 Wendy's 的那个人):
http://scotland-on-rails.s3.amazonaws.com/2A04_DaveThomas-SOR.mp4

看完那个视频后,我买了 Dave Thomas 的整个 series on the Ruby object model来自 Pragmatic,非常值得。

附言任何人都可以随时纠正我忘记的任何事情;就像对象中具体包含什么。

关于ruby - 如何让模块 mixins 为静态方法工作?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4137909/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com