2

我对传球有一个小问题。

def a_method(a, b)
  a + yield(a, b)
end

这工作正常。

k = a_method(1, 2) do |x, y| 
  (x + y) * 3 
end
puts k

但这行不通。

puts a_method(1, 2) do |x, y| 
  (x + y) * 3 
end
# LocalJumpError: no block given (yield)

任何人都可以向我解释一下吗?

谢谢。示例取自 Paolo Perrotta 的 Metaprogramming Ruby。很棒的书。

4

2 回答 2

5

do .. end花括号和花括号的区别在于花括号绑定到最右边的表达式,而do .. end绑定到最左边的表达式。请注意以下示例:

def first(x=nil)
  puts "  first(#{x.inspect}): #{block_given? ? "GOT BLOCK" : "no block"}"
  "f"
end

def second(x=nil)
  puts "    second(#{x.inspect}): #{block_given? ? "GOT BLOCK" : "no block"}"
  "s"
end

first second do |x| :ok end #   second(nil): no block
                            # first("s"): GOT BLOCK

first second {|x| :ok }     #   second(nil): GOT BLOCK
                            # first("s"): no block

在第一种情况下,使用的块do..end将绑定到第一个函数(最左边)。在第二种情况下,用大括号制作的块将绑定到第二个函数(最右边)。

如果您有两个函数和一个块,通常最好使用括号 - 只是为了可读性和避免错误。

puts就像您的问题一样,很容易意外地将块传递给方法。

于 2012-06-06T07:25:40.597 回答
-1

那是因为块被传递给puts而不是传递给a_method.
这应该这样做:

puts (a_method(1, 2) { |x, y| (x + y) * 3 })

# if you want to keep it multilines
puts (a_method(1, 2) { |x, y|
  (x + y) * 3
})
于 2012-06-06T07:11:15.590 回答