web-dev-qa-db-ja.com

Rubyでクラスを異なるパラメーターとデフォルト値で初期化する最も効率的な方法は何ですか?

初期化中に設定したり、デフォルト値を使用したりできるクラスといくつかの属性が必要です。

class Fruit
  attr_accessor :color, :type
  def initialize(color, type)
    @color=color ||= 'green'
    @type=type ||='pear'
  end
end

Apple=Fruit.new(red, Apple)
29
Istvan

この問題を解決する一般的な方法は、デフォルト値を持つハッシュを使用することです。 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

59
Brian Clapper

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

このトピックに関するいくつかの興味深いメモ:

17
knut

ブライアンの答えは素晴らしいですが、ほとんどメタになるようにいくつかの変更を提案したいと思います。

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

もちろん、これはすべてに最適なソリューションではありませんが、コードをより動的にするためのいくつかのアイデアを提供します。

10
Mike Bethany

私はこのようにします:

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もちろん、デフォルト値が存在する場合、デフォルト値を上書きします。

5
vonconrad

より簡単な方法:

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
3
Amer

私は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')
3
aceofbassgreg

さらにおいしい構文糖:

class Fruit
  attr_accessor :color, :type
  def initialize *args
    @color, @type = args 
  end
end

pear = Fruit.new 'green', :pear
1
jasonleonhard