Ruby:在模块中动态定义类方法

时间:2013-07-11 17:54:31

标签: ruby metaprogramming

我在模块中动态定义类方法时遇到了麻烦。见下面的代码。在尝试引用模块中的另一个类方法时,我得到NameError: undefined local variable or method。这似乎可能是一个范围或上下文问题,但到目前为止我还没弄清楚。

module Bar

  def self.included(base)
    base.extend ClassMethods
  end

  module ClassMethods

    def fruits
      ["apple", "orange", "banana"]
    end

    def example_function(string)
      string.upcase
    end

    fruits.each do |fruit|
      method_name = fruit.to_sym
      define_method(method_name) { example_function(fruit) }
    end

  end

end

class Foo
  include Bar
end

puts Foo.apple
puts Foo.orange
puts Foo.banana

我希望能够致电:

puts Foo.apple => "APPLE"
puts Foo.orange => "ORANGE"
puts Foo.banana => "BANANA"

目前,当我尝试其中任何一项时,我收到以下错误: NameError: undefined local variable or method 'fruits' for Bar::ClassMethods:Module

此外,Bar :: ClassMethods中的类方法应该可用于Foo,所以我应该可以调用:

puts Foo.fruits => ["apple", "orange", "banana"]

要求:

  1. 所有代码都在一个模块中。
  2. 该模块允许混合实例和类方法(下文)。
  3. 目标方法是动态定义的。
  4. 阅读“在Ruby中包含vs扩展”(特别是标题为“常用成语”的部分)http://www.railstips.org/blog/archives/2009/05/15/include-vs-extend-in-ruby/

    非常感谢您的帮助!

1 个答案:

答案 0 :(得分:2)

问题是,fruits方法被定义为实例方法,但是您在ClassMethods上没有实例的情况下调用它。只需像这样定义:

def self.fruits
  ["apple", "orange", "banana"]
end

您的代码可以运行。

编辑: 要使fruits方法也可以作为Foo上的类方法访问,您必须以fruits模块内部可访问的方式声明ClassMethods数组。一种方法是将数组声明为类变量,并在fruits方法和each块中使用它。请查看以下代码:

module Bar

  def self.included(base)
    base.extend ClassMethods
  end

  module ClassMethods
    @@fruits = ["apple", "orange", "banana"]

    def fruits
      @@fruits
    end

    def example_function(string)
      string.upcase
    end

    @@fruits.each do |fruit|
      method_name = fruit.to_sym
      define_method(method_name) { example_function(fruit) }
    end
  end
end

class Foo
  include Bar
end

puts Foo.apple
puts Foo.orange
puts Foo.banana
puts Foo.fruits