在
Ruby中,procs似乎可以访问声明它们时出现的局部变量,即使它们在不同的范围内执行:
module Scope1 def self.scope1_method puts "In scope1_method" end end module Scope2 def self.get_proc x = 42 Proc.new do puts x puts self scope1_method end end end Scope1.instance_eval(&Scope2.get_proc)
输出:
42 Scope1 In scope1_method
这是怎么发生的?
解决方法
Proc.new调用为它给出的块创建一个
closure.在为块创建闭包时,块绑定到Proc.new调用范围内的原始变量.
为什么这样做?
它允许Ruby块作为闭包.闭包非常有用,维基百科条目(上面链接)在解释他们的一些应用程序方面表现非常出色.
这是怎么做到的?
这是通过在进入Proc.new方法之前复制存在的Ruby控件框架在Ruby VM(在C代码中)完成的.然后在该控制帧的上下文中运行该块.这有效地复制了此框架中存在的所有绑定.在Ruby 1.8中,您可以在eval.c的proc_alloc函数中找到相应的代码.在Ruby 1.9中,您可以在proc.c中的proc_new函数中找到它.