详解Ruby中的代码块对象Proc
Proc对象
Proc是由块转换来的对象。创建一个Proc共有四种方法,分别是:
示例代码
#法一 inc=Proc.new{|x|x+1} inc.call(2)#=>3 #法二 inc=lambda{|x|x+1} inc.call(2)#=>3 #法三 inc=->(x){x+1} inc.call(2)#=>3 #法四 inc=proc{|x|x+1} inc.call(2)#=>3
除了上面的四种之外,还有一种通过&操作符的方式,将代码块与Proc对象进行转换。如果需要将某个代码块作为参数传递给方法,需要通过为这个参数添加&符号,并且其位置必须是在参数的最后一个
&符号的含义是:这是一个Proc对象,我想把它当成代码块来使用。去掉&符号,将能再次得到一个Proc对象。
示例代码
defmy_method(&the_proc) the_proc end p=my_method{|name|“Hello,#{name}!”} p.class#=>Proc p.call(“Bill”)#=>“Hello,Bill” defmy_method(greeting) “#{greeting},#{yield}!” end my_proc=proc{“Bill”} my_method(“Hello”,&my_proc)
一些需要注意的地方
在使用block时,我会忽略proc的存在,我将proc定位为一个幕后的工作者。我经常写类似下面的代码,
deff(...) ... yield ... end deff(...,&p) ... p.call ... end deff(...,&p) instance_eval&p ... end deff(...,&p) ... defime_methodm,&p ... end
有些新手会写类似下面的一执行就会报错的代码,
deff(...,&p) instance_evalp end deff(...,p) instance_evalp.call end
也有这样写的,
deff(...,&p) instance_evaldo p.call end end
或者
deff(...) instance_evaldo yield end end
我甚至写过类似下面的代码,
deff(...) instance_evalyield end
我们经常在该挂block的时候,却把proc对象当参数传给方法了,或者不明白&p就是block可以直接交给方法使用,我曾经也犯过这样的错误就是因为没有把block和proc正确的区分开来,&p是block,p是proc,不到万不得已的情况下不要显式地创建proc,每当我对block和proc之间的关系犯糊涂时,我就会念上几句。