Динамически создавать атрибуты класса с помощью attr_accessor - PullRequest
9 голосов
/ 03 ноября 2010

В Ruby есть способ динамически добавить переменную экземпляра в класс? Например:

class MyClass

  def initialize
    create_attribute("name")
  end 

  def create_attribute(name)
    attr_accessor name.to_sym
  end

end

o =  MyClass.new
o.name = "Bob"
o.name

Ответы [ 3 ]

26 голосов
/ 03 ноября 2010

Один способ (есть и другие) - использовать instance_variable_set и instance_variable_get следующим образом:

class Test
    def create_method( name, &block )
        self.class.send( :define_method, name, &block )
    end

    def create_attr( name )
        create_method( "#{name}=".to_sym ) { |val| 
            instance_variable_set( "@" + name, val)
        }

        create_method( name.to_sym ) { 
            instance_variable_get( "@" + name ) 
        }
    end
end

t = Test.new
t.create_attr( "bob" )
t.bob = "hello"
puts t.bob
1 голос
/ 03 ноября 2010

может быть,

instance_variable_set(name,value)

это то, что вы хотите!

например:

class Mclass
  def show_variables
    puts self.class.instance_variables
  end
end

Mclass.instance_variable_set(:@test,1)
o=Mclass.new
o.show_variables

знаете, класс тоже объект.

0 голосов
/ 08 июня 2016

Запутался, отвечая на комментарий / вопрос @ ericso в другом комментарии, так что вот он в ответе - модуль, который я использую (основываясь на коде @ ReeseMoore), чтобы сделать это и при необходимости установить переменные экземпляра: *

# adds ability to dynamically create instance vars & attr_accessors
module Attributable
  def create_method(name, &block)
    self.class.send(:define_method, name.to_sym, &block)
  end

  def create_setter(m)
    create_method("#{m}=".to_sym) { |v| instance_variable_set("@#{m}", v) }
  end

  def create_getter(m)
    create_method(m.to_sym) { instance_variable_get("@#{m}") }
  end

  def set_attr(method, value)
    create_setter method
    send "#{method}=".to_sym, value
    create_getter method
  end
end
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...