本文共 2615 字,大约阅读时间需要 8 分钟。
今天在书上看到这么一个例子,虽然我多多少少对 Ruby 的 block 有一些了解,但是第一次见到下面这种时,还是有点懵的:
# Rubydef block_args_test yield() yield(1) yield(1, 2, 3)endputs "通过 |a| 接收块变量"block_args_test do |a| p [a]endputsputs "通过 |a, b, c| 接收块变量"block_args_test do |a, b, c| p [a, b, c]endputsputs "通过 |*a| 接收块变量"block_args_test do |*a| p [a]endputs------------------------------------------------------------通过 |a| 接收块变量[nil][1][1]通过 |a, b, c| 接收块变量[nil, nil, nil][1, nil, nil][1, 2, 3]通过 |*a| 接收块变量[[]][[1]][[1, 2, 3]]
所以,为了啃下这块硬骨头,先从一个简单的例子讲起吧!
# Ruby# 定义块def myloop while true yield # 执行块 endendnum = 1myloop do puts "num is #{num}" break if num > 10 num *= 2end------------------------------------------------------------num is 1num is 2num is 4num is 8num is 16
首先是定义块,在定义好块以后,使用块的 do ~ end 语句时,其实质是把块里边的代码:
puts "num is #{num}" break if num > 10 num *= 2
放到了 yield 的地方去执行,所以这儿是比较好理解的。
我们把上面的简单例子稍微改造一下,改成可以传参给 block,下面还是可以得到同上面一样的输出。
# Ruby# 定义块def myloop while true yield(2) # 执行块 endendnum = 1myloop do |i| puts "num is #{num}" break if num > 10 num *= iend------------------------------------------------------------num is 1num is 2num is 4num is 8num is 16
虽然比不传参的稍微复杂一点,但是我相信大家还是能看懂的。
现在,我们再复杂化一点,块定义的时候设定两个参数,但是 block 执行的时候只给块传一个参数。
# Ruby# 定义块def myloop while true yield(2, 3) # 执行块 endendnum = 1myloop do |i| puts "num is #{num}" break if num > 10 num *= iend------------------------------------------------------------num is 1num is 2num is 4num is 8num is 16
还是一样的结果,似乎多的参数对执行结果并没有什么影响。到现在,我们基本上能把最开始的例子解释一下了。
# Rubydef block_args_test yield() yield(1) yield(1, 2, 3)endputs "通过 |a| 接收块变量"block_args_test do |a| p [a]endputsputs "通过 |a, b, c| 接收块变量"block_args_test do |a, b, c| p [a, b, c]endputsputs "通过 |*a| 接收块变量"block_args_test do |*a| p [a]endputs------------------------------------------------------------通过 |a| 接收块变量[nil] # yield() 没传递参数,所以 a 没接收到参数,a 为 nil[1] # yield(1) 传递一个参数,所以 a 接收到传递的参数,a 为 1[1] # yield(1, 2, 3) 传递三个参数,但是只有一个块变量接收,所以接收到第一个参数,a 为 1通过 |a, b, c| 接收块变量[nil, nil, nil] # yield() 没传递参数,所以三个块变量 a, b, c 均为 nil [1, nil, nil] # yield() 传递一个参数,所以 a 接收到传递的参数,a 为 1,剩下 2 个块变量 b, c 均为 nil[1, 2, 3] # yield() 传递三个个参数,a, b, c 三个块变量分别接收一个通过 |*a| 接收块变量# *a 是匹配任意个的意思,所以无论 yield 传递过来多少参数,块变量都会将多个参数变成一个数组# 我们可以来个下面的简单例子印证一下# [20] pry(main)> def test(*args)# [20] pry(main)* p [args] # [20] pry(main)* end # => :test# [21] pry(main)> test(1, 2, 3)# [[1, 2, 3]]# => [[1, 2, 3]][[]] # yield() 没传递参数,所以 a 为 [][[1]] # yield() 传递一个参数,所以 a 为 [1][[1, 2, 3]] # yield() 传递三个个参数,匹配三个参数以后,所以 a 为 [1, 2, 3]
转载地址:http://djjqi.baihongyu.com/