我正在尝试从散列(使用嵌套散列)生成attr_reader,以便它自动镜像instance_variable创建。
这是我到目前为止所拥有的:
data = {:@datetime => '2011-11-23', :@duration => '90', :@class => {:@price => '£7', :@level => 'all'}}
class Event
#attr_reader :datetime, :duration, :class, :price, :level
def init(data, recursion)
data.each do |name, value|
if value.is_a? Hash
init(value, recursion+1)
else
instance_variable_set(name, value)
#bit missing: attr_accessor name.to_sym
end
end
end
但是我找不到办法做到这一点:(
答案 0 :(得分:35)
您需要在attr_accessor
类上调用(私有)类方法Event
:
self.class.send(:attr_accessor, name)
我建议您在此行添加@
:
instance_variable_set("@#{name}", value)
不要在哈希中使用它们。
data = {:datetime => '2011-11-23', :duration => '90', :class => {:price => '£7', :level => 'all'}}
答案 1 :(得分:3)
你可以使用method_missing:
来做一些元魔法来解决这个问题class Event
def method_missing(method_name, *args, &block)
if instance_variable_names.include? "@#{method_name}"
instance_variable_get "@#{method_name}"
else
super
end
end
end
这将允许通过object.variable语法访问对象实例变量,如果对象定义了这些变量,而不需要通过attr_accessor修改整个类。
答案 2 :(得分:1)
attr_accessor
是一个类方法,因此需要在类上调用。它也是一个私有方法,因此您需要在类对象为self
的上下文中调用它。
举个例子:
class C
def foo
self.class.instance_eval do
attr_accessor :baz
end
end
end
在该实例上创建C
的实例并调用foo
之后,该实例以及所有未来的实例将包含方法baz
和baz=
。< / p>