1

我创建了一个对象调用Proc.new并将一个块作为参数传递给它:

a = Proc.new{|x| x = x*10; puts(x)}
#=> #<Proc:0xd26fd8@(irb):3>
a.call(10)
#100
#=> nil
a.call(10,20)
#100
#=> nil
a.call(10,20,40)
#100
#=> nil

我也没用过splat operator(*)。但是如何阻止参数x能够忽略额外的参数呢?

当我们做同样的事情时,我们会得到一个明确的错误,但为什么块参数不是这种情况?

def show(x)
print "X::#{x}"
end
#=> nil
show(10)
#X::10#=> nil
show(10,20)
#ArgumentError: wrong number of arguments (2 for 1)
#        from (irb):6:in `show'
#        from (irb):10
#        from C:/Ruby193/bin/irb:12:in `<main>'
4

2 回答 2

2

Procs将缺少的参数转换为 nil 而lambda没有。

如果您想容忍错误,请使用 Procs。否则你会想去lambda

l = ->(x) { x = x * 10; puts x }
=> #<Proc:0x007fada3be9468@(pry):12 (lambda)>
l.call(10, 20)
ArgumentError: wrong number of arguments (2 for 1)
from (pry):12:in `block in <main>'
于 2013-03-11T14:45:37.277 回答
1

这就是 Procs 的工作方式,在内部,他们不在乎是否传递了太多参数。

Proc#call将接受一个参数数组并将它们绑定到块的参数,如果计数不匹配,则不会抱怨。

但是,Proc Lambdas 会抱怨它,这是它们与常规 Procs 之间的区别之一:

2.0.0p0 :006 > r = lambda { |x| puts x }
 => #<Proc:0x007fac6913b600@(irb):6 (lambda)>
2.0.0p0 :007 > r.call(1,2)
ArgumentError: wrong number of arguments (2 for 1)
from (irb):6:in `block in irb_binding'
from (irb):7:in `call'
from (irb):7
from /Users/Intrepidd/.rvm/rubies/ruby-2.0.0-p0/bin/irb:16:in `<main>'
于 2013-03-11T14:36:55.257 回答