Ответ 1
Вы можете использовать instance_variable_set
следующим образом:
params.each do |key, value|
self.instance_variable_set("@#{key}".to_sym, value)
end
в 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
Вы можете использовать instance_variable_set
следующим образом:
params.each do |key, value|
self.instance_variable_set("@#{key}".to_sym, value)
end
Чтобы все было просто:
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
Вы могли бы сделать что-то умнее с поворотом метапрограммирования, но действительно ли это необходимо?
Я предлагаю
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
Я думаю, вы могли бы просто поставить:
Weekend < Struct.new(:start_date, :end_date, :title, :description, :location)
Затем добавьте еще что-нибудь в класс выходных дней:
class Weekend
#whatever you need to add here
end
Иногда 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