以前觉得Ruby中的闭包是个很神秘的东西,这两天一直在好好研究,终于明白了是怎么回事。下面贴出我的研究成果,其参考了李刚的Ruby on Rails最佳开发实践 那本书。
所谓闭包,就是当一个方法返回时,该方法还没有释放资源的栈区。对于C或者Java等其他主流语言,当一个函数或方法返回后,所有的局部变量将不可访问,因为它们所在的内存栈已经被销毁。但在Ruby里,如果该方法返回一个代码块,而该代码块获得了局部变量的引用,则这些局部变量将在方法返回后依然可以访问。
如下代码示范了当一个方法返回后,该方法内的局部变量依然可以访问的效果。
# 定义一个方法
def show(name)
#局部变量
count = 1
# 该方法返回一个Proc对象,该Proc对象获得了局部变量name的引用
return proc do |value|
# 代码块访问count局部变量
puts count += 1
# 代码块访问name局部变量
puts name + " 代码块里执行的动作 " + value
end
end
# 调用show方法
sh = show("Ruby")
# 如果没有闭包,当show方法调用结束,则show方法中的
# name和count两个局部变量都被释放,不可再访问
# 但实际上,因为show访问返回的了Proc对象,该对象
# 引用了name和count两个局部变量,因此形成了闭包,从而
# 允许在show方法外通过该Proc对象来访问这两个局部变量
sh.call("代码块参数")
sh.call("代码块参数")
从上面的程序中不难看出,当执行了show("Ruby")代码后,该方法返回的Proc对象被赋给了sh对象,这就形成了一个闭包,从而允许show方法内的局部变量可以通过Proc对象来访问。
除此之外,还可以直接将一个方法定义成一个代码块。一旦将一个方法定义成一个代码块,就可以将该方法当成代码块使用。看如下方法定义:
# 定义一个into方法,该方法返回一个Proc对象
def into(anArray)
# 直接返回一个代码块转换的Proc对象
# 代码块访问了anArray局部变量,则该变量可以在into方法外访问
return proc do |val|
anArray << val
end
end
上面的into方法返回一个使用proc执行代码块后的结果,也就是一个Proc对象,一样会形成一个闭包,从而允许在into方法返回后依然可以访问局部变量anArray。
下面使用该into方法,如果希望把into方法的返回值当成代码块,应该在该方法前增加“&”符号。看如下代码:
# 因为调用into方法时会创建一个闭包
# 从而允许在into方法执行结束后访问a变量
["Ruby" , "Rails" , "RoR"].each( &into( a = [] ) )
p a
运行上面的程序,看到如下结果:
["Ruby", "Rails", "RoR"]