14

我想做以下事情:

我想声明一个遍历字典的类的实例变量。

假设我有这个哈希

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

我知道这行不通!我只是放了这段代码,看看你是否能更清楚地理解我想要什么。

谢谢!

4

4 回答 4

28
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 是一个只属于单个对象的特殊类,因此定义的方法将是该对象的实例方法,但不属于该对象的普通类的其他实例。

于 2009-10-23T18:35:26.047 回答
4

Chuck 的回答比我最近两次尝试的要好。eigenclassself.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
于 2009-10-23T18:35:50.790 回答
4
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
于 2009-10-23T18:36:15.387 回答
2

http://facets.rubyforge.org/apidoc/api/more/classes/OpenStructable.html

OpensStructable 是一个 mixin 模块,可以为任何类或对象提供 OpenStruct 行为。OpenStructable 允许扩展具有任意属性的数据对象。

于 2009-10-23T20:44:36.537 回答