1、什么是代码块
在Ruby中,{}或do…end之间的代码是一个代码块。代码块只能出现在一个方法的后边,它紧接在方法最后一个参数的同一行上,由yield关键字调用。例如:
[1,2,3,4,5].each do |i|
puts i
end
块变量:以yield关键字调用block也可以传递参数,block中竖线(|)之间给出的参数名用于接收来自yield的参数。
竖线之间(如上例中的 | i |)的变量被称作块变量,作用和一个正常方法的参数一样
2、掌握代码块的编写
最常见、最简单、最富争议、最有Ruby风格的方式是blocks。写法如下:
array.collect! do |n|
n ** 2
end
puts array.inspect
#=> [1, 4, 9, 16]
do…end构成一个block。然后把这个block通过collect!传给一个数组。就可以使用block中的n来迭代数组中每个元素。
collect!是Ruby库里的方法,下面我们来写一个自己的类似方法iterate!
def iterate!
self.each_with_index do |n, i|
self[i]=yield(n)
end
end
end
array=[1, 2, 3, 4]
array.iterate! do |n|
n ** 2
end
puts array.inspect
#=> [1, 4, 9, 16]
首先,我们打开Array,并添加进iterate!方法。方法名以!结尾表示危险方法,引起注意。现在我们就可能像使用collect!一样使用iterate!
与属性不同,在方法中不需要指定block的名字,而是使用yield来调用。yield会执行block中的代码。同时,注意我们是怎么把n(each_with_index当前处理的数字)传给yield的。传给yield的参数即对应了block中的参数(||中的部分)。现在n就能被block调用并在yield调用中返回n**2。
整个调用如下:
1、一个整数组成的数组调用iterate!
2、当yield被调用时,把n(第一次为1,第二次为2,…)传给block
3、block对n进行n**2。因为是最后一行,自动作为结果返回。
4、yield得到block的结果,并把值重写到array里。
5、数据中每个对象执行相同操作。
3、{}和do…end优先级不同
在传递一个block时,使用{}传递的block比使用do…end的优先级要高;
为了避免引起歧义,最好使用()将参数括起来。例如:
puts x
end
是正确的,但是 1.upto 3 {|x| puts x} 编译不通过,应该写成 1.upto(3) {|x| puts x}
原因:
1.upto 3 do…end 中block会传递到upto方法里面,3会作为一个参数传递给upto
1.upto 3 {|x| puts x} 一句会把3当做函数名,将block传递到这个函数,其返回值作为upto方法的参数,所以编译不过,需加()。