Подтвердить что ты не робот

В Ruby я могу автоматически заполнить переменные экземпляра каким-то образом в методе initialize?

в Ruby я могу автоматически заполнить переменные экземпляра каким-то образом в методе initialize?

Например, если у меня было:

class Weekend
  attr_accessor :start_date, :end_date, :title, :description, :location

  def initialize(params)
    # SOMETHING HERE TO AUTO POPULATE INSTANCE VARIABLES WITH APPROPRIATE PARAMS
  end

end
4b9b3361

Ответ 1

Вы можете использовать instance_variable_set следующим образом:

params.each do |key, value|
  self.instance_variable_set("@#{key}".to_sym, value)
end

Ответ 2

Чтобы все было просто:

class Weekend
  attr_accessor :start_date, :end_date, :title, :description, :location

  def initialize(params)
    @start_date = params[:start_date] # I don't really know the structure of params but you have the idea
    @end_date   = params[:end_date]
  end
end

Вы могли бы сделать что-то умнее с поворотом метапрограммирования, но действительно ли это необходимо?

Ответ 3

Я предлагаю

class Weekend
  @@available_attributes = [:start_date, :end_date, :title, :description, :location]
  attr_accessor *@@available_attributes

  def initialize(params)
    params.each do |key,value|
      self.send(:"#{key}=",value) if @@available_attributes.include?(key.to_sym)
    end
  end
end

Ответ 4

Я думаю, вы могли бы просто поставить:

Weekend < Struct.new(:start_date, :end_date, :title, :description, :location)

Затем добавьте еще что-нибудь в класс выходных дней:

class Weekend
#whatever you need to add here
end

Ответ 5

Иногда Ruby может быть пугающим. Нет петли в поле зрения!

class Weekend < Struct.new(:start_date, :end_date, :title, :description, :location)
  # params: Hash with symbols as keys
  def initialize(params)
    # arg splatting to the rescue
    super( * params.values_at( * self.class.members ) )
  end
end

Обратите внимание, что вам даже не нужно использовать наследование - во время создания можно настроить новый Struct:

Weekend = Struct.new(:start_date, :end_date, :title, :description, :location) do
  def initialize(params)
    # same as above
  end
end

Тест:

weekend = Weekend.new(
  :start_date => 'start_date value',
  :end_date => 'end_date value',
  :title => 'title value',
  :description => 'description value',
  :location => 'location value'
)

p [:start_date , weekend.start_date  ]
p [:end_date   , weekend.end_date    ]
p [:title      , weekend.title       ]
p [:description, weekend.description ]
p [:location   , weekend.location    ]

Обратите внимание, что это фактически не задает переменные экземпляра. У вас класс будут непрозрачные геттеры и сеттеры. Если вы не хотите их раскрывать, вы можете обернуть вокруг него еще один класс. Вот пример:

# this gives you more control over readers/writers
require 'forwardable'
class Weekend
  MyStruct = ::Struct.new(:start_date, :end_date, :title, :description, :location)
  extend Forwardable
  # only set up readers
  def_delegators :@struct, *MyStruct.members

  # params: Hash with symbols as keys
  def initialize(params)
    # arg splatting to the rescue
    @struct = MyStruct.new( * params.values_at( * MyStruct.members ) )
  end
end