1

我在 Ruby Monk 上遇到了一个问题,https://rubymonk.com/learning/books/1-ruby-primer/problems/155-restaurant#solution4804

他们的解决方案很棒;我喜欢它,它比我的更紧凑。问题是我的,我只是不明白为什么它只有在我从 cost 参数中删除 splat 运算符时才有效orders。即使我不应该这样做,我也在努力弄清楚发生了什么。我知道有时没有必要了解一切,最好继续前进……但好奇。

这是我的:

class Restaurant
  def initialize(menu)
    @menu = menu
  end

  def cost(*orders)
    total_cost = 0
    orders.each do |item, number|
    total_cost += @menu[item] * number
  end
end

menu = {:rice => 3, :noodles => 2}
orders = {:rice => 1, :noodles => 1}
eat = Restaurant.new(menu)
puts eat.cost(orders)

编辑:在下面包括他们建议的解决方案

class Restaurant
  def initialize(menu)
    @menu = menu
  end

  def cost(*orders)
    orders.inject(0) do |total_cost, order|
      total_cost + order.keys.inject(0) {|cost, key| cost + @menu[key]*order[key] }
    end
  end
end

编辑:在评论中澄清并回答我自己的问题

我尝试了这些实验,它显示inject“删除”了“穿上”的数组括号。也许不是最合适的思考方式?它确实有助于消除我的困惑。

order = { :rice => 1, :noodles => 1 }
menu = { :rice => 3, :noodles => 2 }

[order].inject(0) do |bla, blu|
    p bla       #=> 0
    p blu       #=> {:rice=>1, :noodles=>1}
    p blu.keys  #=> [:rice, :noodles]
end
4

1 回答 1

3

当你写:

def cost(*orders)
end

然后传递给该cost方法的所有参数将被放入一个名为orders. 因此,这两个是等价的:

def cost(*orders)
  p orders.class #=> Array
  p orders       #=> [1,2,3]
end
cost(1,2,3)

def cost(orders)
  p orders.class #=> Array
  p orders       #=> [1,2,3]
end
cost( [1,2,3] )     # note the array literal brackets

在您的情况下,当您删除“splat”时,您是在说“设置orders为引用直接传入的任何内容”。在这种情况下,您将传递给它一个哈希,并且当您迭代一个哈希时,您将获得每个条目的键/值对。这正是你想要的。

但是,当您确实遇到问题时,您会得到以下信息:

def cost(*orders)
  p orders.class #=> Array
  p orders       #=> [{:rice=>1, :noodles=>1}]
end
orders = {:rice=>1, :noodles=>1}
cost(orders)

因此,您将散列包装在一个数组中,然后迭代数组的元素。因此,传递给块的第一个值是整个哈希,并且没有第二个参数。

def cost(*orders)
  p orders.class #=> Array
  p orders       #=> [{:rice=>1, :noodles=>1}]
  orders.each do |item,number|
    p item       #=> {:rice=>1, :noodles=>1}
    p number     #=> nil
  end
end
orders = {:rice=>1, :noodles=>1}
cost(orders)

在这一点上,你不能乘以任何东西nil,所以你的代码会中断。

于 2014-02-13T03:23:21.470 回答