初期化中に設定したり、デフォルト値を使用したりできるクラスといくつかの属性が必要です。
class Fruit
attr_accessor :color, :type
def initialize(color, type)
@color=color ||= 'green'
@type=type ||='pear'
end
end
Apple=Fruit.new(red, Apple)
この問題を解決する一般的な方法は、デフォルト値を持つハッシュを使用することです。 Rubyは、ハッシュがメソッドの最後のパラメータである場合、ハッシュ値を渡すための適切な構文を持っています。
class Fruit
attr_accessor :color, :type
def initialize(params = {})
@color = params.fetch(:color, 'green')
@type = params.fetch(:type, 'pear')
end
def to_s
"#{color} #{type}"
end
end
puts(Fruit.new) # prints: green pear
puts(Fruit.new(:color => 'red', :type => 'grape')) # prints: red grape
puts(Fruit.new(:type => 'pomegranate')) # prints: green pomegranate
良い概要はこちらです http://deepfall.blogspot.com/2008/08/named-parameters-in-Ruby.html
Ruby 2.0以降、名前付きパラメータまたはキーワードパラメータがサポートされています。
以下を使用できます。
class Fruit
attr_reader :color, :type
def initialize(color: 'green', type: 'pear')
@color = color
@type = type
end
def to_s
"#{color} #{type}"
end
end
puts(Fruit.new) # prints: green pear
puts(Fruit.new(:color => 'red', :type => 'grape')) # prints: red grape
puts(Fruit.new(:type => 'pomegranate')) # prints: green pomegranate
このトピックに関するいくつかの興味深いメモ:
ブライアンの答えは素晴らしいですが、ほとんどメタになるようにいくつかの変更を提案したいと思います。
class Fruit
# Now this is the only thing you have to touch when adding defaults or properties
def set_defaults
@color ||= 'green'
@type ||= 'pear'
end
def initialize(params = {})
params.each { |key,value| instance_variable_set("@#{key}", value) }
set_defaults
instance_variables.each {|var| self.class.send(:attr_accessor, var.to_s.delete('@'))}
end
def to_s
instance_variables.inject("") {|vars, var| vars += "#{var}: #{instance_variable_get(var)}; "}
end
end
puts Fruit.new
puts Fruit.new :color => 'red', :type => 'grape'
puts Fruit.new :type => 'pomegranate'
puts Fruit.new :cost => 20.21
puts Fruit.new :foo => "bar"
f = Fruit.new :potato => "salad"
puts "f.cost.nil? #{f.cost.nil?}"
どの出力:
@color: green; @type: pear;
@color: red; @type: grape;
@color: green; @type: pomegranate;
@color: green; @type: pear; @cost: 20.21;
@color: green; @type: pear; @foo: bar;
f.cost.nil? true
もちろん、これはすべてに最適なソリューションではありませんが、コードをより動的にするためのいくつかのアイデアを提供します。
私はこのようにします:
class Fruit
attr_accessor :color, :type
def initialize(args={})
options = {:color => 'green', :type => 'pear'}.merge(args)
self.color = options[:color]
self.type = options[:type]
end
end
Apple = Fruit.new(:color => 'red', :type => 'Apple')
このようにして、欠落している引数やその順序について心配する必要はありません。デフォルトの値が常にそこにあります。 .merge
もちろん、デフォルト値が存在する場合、デフォルト値を上書きします。
より簡単な方法:
class Fruit
attr_accessor :color, :type
def initialize(color = 'green', type = 'pear')
@color = color
@type = type
end
def to_s
"#{color} #{type}"
end
end
puts Fruit.new # prints: green pear
puts Fruit.new('red','Apple') # prints: red Apple
puts Fruit.new(nil,'pomegranate') # prints: green pomegranate
私はvonconradの答えが好きですが、別のdefaults
メソッドがあります。コード行の点では効率的ではないかもしれませんが、意図を明らかにし、認知オーバーヘッドが少なく、認知オーバーヘッドが少ないほど、開発の効率が高くなります。
class Fruit
attr_accessor :color, :type
def initialize(args={})
options = defaults.merge(args)
@color = options.fetch(:color)
@type = options.fetch(:type)
end
def defaults
{
color: 'green',
type: 'pear'
}
end
end
Apple = Fruit.new(:color => 'red', :type => 'Apple')
class Fruit
attr_accessor :color, :type
def initialize *args
@color, @type = args
end
end
pear = Fruit.new 'green', :pear