深入理解Ruby中的代码块block特性

所属分类: 脚本专栏 / ruby专题 阅读数: 827
收藏 0 赞 0 分享

block是什么?

在Ruby中,block并不罕见。官方对block的定义是“一段被包裹着的代码”。当然,我觉得这样的解释不会让你变的更明白。

对block的一种更简单的描述是“一个block就是一段存储在一个变量中的代码,它和其他的对象一样,可以被随时的运行”

然后,咱们通过看一些代码,之后再把这些代码重构成Ruby中的block形式。通过代码来实际的感受,更加直观。

比如,对两个数做加法?

puts 5 + 6
# => 11

嗯,这样写是可以的。但是,这样的代码只做到了block定义的前半部分——它是一段代码。但是它并没有“被包裹起来”,也没有“存储在一个变量中”。

所以,我们需要继续修改。不过在把它包裹起来之前,我们先改进一下,让它看起来更通用。

a = 5
b = 6
puts a + b
# => 11

好~这样就可以了——我们用变量替换了之前的数字。这段代码执行了一个相加的过程,但是,它仍然没有被储存在一个变量中。

现在,咱们来实现它。

addition = lambda { |a, b| return a+b }
puts addition.call(5, 6)
# => 11

好啦,现在你把它很好的包裹起来了——这就是一个block!

使用‘lambda'关键字,是Ruby中创建block的最常见的方法。还有其他的方法也可以做到,不过现在先不管其他的方法。

这个时候你可能会想“等等,这玩意儿看起来就像是一个方法(method),除了没有类和对象“。你说的没错。甚至可以这样去理解:一个block就像一个方法(method),但是它不与任何的对象关联。

咱们继续,更仔细的来看看block。

一个块包含的代码块。你可以分配一个名称,一个块。 块中的代码总是被括在大括号里({})或是do...end里。

[1, 2, 3].each do |i|
 puts i
end

#=> 1
  2
  3

上面这个例子, each方法后面加一个do...end结构,那就是一个块。

Ruby中任何一个方法你都可以传递一个块。

  def test;end
  test{ puts i}

def test
  yield
 end
 test{puts "hello test!"}

 def test(x)
  yield(x)
 end
 test('world!'){|x| puts "hello #{x}"}

yield关键字不仅可以挂载块(block)代码,而且可以给块传递参数。

def test(&block)
  block.call("world")
 end

 test{|msg| puts "hello #{msg}"}
block到了方法内部,已经被&转化为了一个Proc对象。

 def test(&block)
  inner_test(&block)
 end

 def inner_test
  yield("haha!")
 end

 test{|msg| puts "hello #{msg}"}

test方法传进去的block被转化为了Proc对象,而其内部的inner_test又利用「&」把这个Proc对象转化为了块(block)

block是对象吗?当然,就像Ruby中的其它东西一样,block也是对象。

empty_block = lambda { }
puts empty_block.object_id
# => 28765760
puts empty_block.class
# => Proc
puts empty_block.class.superclass
# => Object

如你所见,我们创建的这个block有一个 object_id ,属于 Proc 类(这是Ruby里面对一个block的称呼),而这个类本身就是 Object 的子类。

我们甚至可以反过来,从block定义方法(method)。一个方法(method)就是绑定了一个对象的block,从而可以访问对象的“状态”。

下面我来演示一下逆向的用一个方法(method)来创建一个block。有一些更传统的方法来实现前面的问题(同时请原谅我糟糕的对象建模)

class Calculator
 def add(a, b)
  return a+b
 end
end

puts Calculator.new.add(5, 6)
# => 11

这段代码当然能够很好的工作。然后,做一点修改。

class Calculator
 def add(a, b)
  return a+b
 end
end

addition_method = Calculator.new.method("add")
addition = addition_method.to_proc

puts addition.call(5, 6)
# => 11

现在呢,你就把一个传统的方法(method)转换为了一个block!

block化你的代码!

咱们来构造4个block,分别用来进行加减乘除的运算。每个block应该接受两个值作为变量,然后执行操作并返回结果。

Addition = lambda { |a, b| return a+b }

Subtraction = lambda { |a, b| return a-b }

Multiplication = lambda { |a, b| return a*b }

Division = lambda { |a, b| return a/b }

# 使用的时候通过call来使用
Addition.call(5, 6)
# => 11

更多精彩内容其他人还在看

Rails link_to 详解

想学习rauks link_to的朋友可以参考下面的例子。
收藏 0 赞 0 分享

ruby 小脚本搞定CVS服务器更换后checkout下来的工程迁移

CVS换了新的服务器,原来的工程需要更改Server配置,这个东东手工做起来 可是个体力活,写了一个脚本分发下来。
收藏 0 赞 0 分享

Ruby 魔法 学习笔记之一

Ruby的许多动态特性,让Ruby具有很多魔法,这个魔法足以让你来定制你自己的语言DSL, Rails就是Ruby在Web的DSL.
收藏 0 赞 0 分享

Ruby self在不同环境的含义

Ruby的self在不同的环境中有不同的含义,这点和java的this不同,原因是java实际上只有一种环境--在class的实例方法定义中使用,代表访问这个方法参数自动传进的那个对象。
收藏 0 赞 0 分享

ruby 程序的执行顺序

ruby程序的执行是顺序执行的,他是从脚本的第一行执行到最后一行,但是实际执行顺序是
收藏 0 赞 0 分享

ruby on rails 代码技巧

对于rails的一些使用技巧的代码
收藏 0 赞 0 分享

ruby 标准类型总结

诠释分析了ruby的标准类型,学习ruby的朋友,需要了解和掌握的。
收藏 0 赞 0 分享

ruby 去掉文件里重复的行

以前合并后台字典时,有重复的都是用vbs去,最近又看了一天的ruby,想起来写一下,没想到代码如此精简
收藏 0 赞 0 分享

Ruby rails 页面跳转(render和redirect_to)

今天在做R.R.log的时候发现个问题,在修改密码的时候如果没有通过校验,没有显示校验错误的信息。
收藏 0 赞 0 分享

Ruby 取得指定月日期数的方法

取得指定月日期数的Ruby代码
收藏 0 赞 0 分享
查看更多