将块传递给方法 – Ruby

参见英文答案 > Code block passed to each works with brackets but not with ‘do’-‘end’ (ruby)3个
我对传球有一点疑问.
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,y| 
  (x + y) * 3 
end
# LocalJumpError: no block given (yield)

任何人都可以向我解释这个吗?

谢谢. Paolo Perrotta从Metaprogramming Ruby获取的示例.好书.

解决方法

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方法.

相关文章

以下代码导致我的问题: class Foo def initialize(n=0) @n = n end attr_accessor :n d...
这是我的spec文件,当为上下文添加测试“而不是可单独更新用户余额”时,我得到以下错误. require 's...
我有一个拦截器:DevelopmentMailInterceptor和一个启动拦截器的inititializer setup_mail.rb. 但我想将...
例如,如果我有YAML文件 en: questions: new: 'New Question' other: recent: ...
我听说在RSpec中避免它,let,let !,指定,之前和主题是最佳做法. 关于让,让!之前,如果不使用这些,我该如...
我在Rails中使用MongoDB和mongo_mapper gem,项目足够大.有什么办法可以将数据从Mongoid迁移到 Postgres...