我要做到以下几点:
我想在一本字典来声明一个类迭代的实例变量。
让我们假设我有这个哈希
hash = {"key1" => "value1","key2" => "value2","key3" => "value3"}
我想有每个按键为一个类的实例变量。 我想知道如果我能声明变量遍历散列。 事情是这样的:
class MyClass
def initialize()
hash = {"key1" => "value1","key2" => "value2","key3" => "value3"}
hash.each do |k,v|
@k = v
end
end
end
我知道这行不通! 我只能把这段代码,看看,如果你能明白我想更清楚。
谢谢!
class MyClass
def initialize()
hash = {"key1" => "value1","key2" => "value2","key3" => "value3"}
hash.each do |k,v|
instance_variable_set("@#{k}",v)
# if you want accessors:
eigenclass = class<<self; self; end
eigenclass.class_eval do
attr_accessor k
end
end
end
end
该eigenclass只是属于单个对象的特殊类,所以定义的方法会出现该对象的实例方法,但不属于该对象的正常类的其他实例。
class MyClass
def initialize
# define a hash and then
hash.each do |k,v|
# attr_accessor k # optional
instance_variable_set(:"@#{k}", v)
end
end
end
Chuck的答案是比我过去两年的尝试更好。 该eigenclass不self.class
像我已经想到了; 它采取了更好的测试比我曾写信给意识到这一点。
用我的旧代码,我在下面的方式进行测试,发现班级确实操纵,而不是实例:
a = MyClass.new :my_attr => 3
b = MyClass.new :my_other_attr => 4
puts "Common methods between a & b:"
c = (a.public_methods | b.public_methods).select { |v| a.respond_to?(v) && b.respond_to?(v) && !Object.respond_to?(v) }
c.each { |v| puts " #{v}" }
输出是:
Common methods between a & b:
my_other_attr=
my_attr
my_attr=
my_other_attr
这清楚地反驳我的前提。 我的道歉查克,你是对所有沿。
年长的答案:
attr_accessor
仅在一个类定义,而不是一个实例的初始化时评估工作。 因此,直接做你想做的唯一方法是使用instance_eval
一个字符串:
class MyClass
def initialize(params)
#hash = {"key1" => "value1","key2" => "value2","key3" => "value3"}
params.each do |k,v|
instance_variable_set("@#{k}", v)
instance_eval %{
def #{k}
instance_variable_get("@#{k}")
end
def #{k}= (new_val)
instance_variable_set("@#{k}", new_val)
end
}
end
end
end
为了检验这一尝试:
c = MyClass.new :my_var => 1
puts c.my_var
http://facets.rubyforge.org/apidoc/api/more/classes/OpenStructable.html
OpensStructable是一个混合模块,其可以提供OpenStruct行为的任何类或对象。 OpenStructable允许任意属性的数据对象的推广。