Объявление переменных экземпляра итерацией по хешу! - PullRequest
14 голосов
/ 23 октября 2009

Я хочу сделать следующее:

Я хочу объявить переменные экземпляра класса, повторяющиеся по словарю.

Давайте предположим, что у меня есть этот хэш

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 ]

28 голосов
/ 23 октября 2009
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

Собственный класс - это специальный класс, принадлежащий только одному объекту, поэтому методы, определенные там, будут методами экземпляра этого объекта, но не принадлежат другим экземплярам нормального класса объекта.

4 голосов
/ 23 октября 2009
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
4 голосов
/ 23 октября 2009

Ответ Чака лучше, чем мои последние две попытки. Собственный класс не 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
2 голосов
/ 24 октября 2009

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

OpensStructable - это смешанный модуль который может обеспечить поведение OpenStruct к любому классу или объекту. OpenStructable позволяет расширение объектов данных с произвольные атрибуты.

...