我在模块中动态定义类方法时遇到了麻烦。见下面的代码。在尝试引用模块中的另一个类方法时,我得到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"]
要求:
阅读“在Ruby中包含vs扩展”(特别是标题为“常用成语”的部分)http://www.railstips.org/blog/archives/2009/05/15/include-vs-extend-in-ruby/
非常感谢您的帮助!
答案 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