ruby中没有嵌套的作用域,它的作用域之间是分开的,一旦进入一个新作用域,原先的绑定就会被替换为一组新的绑定。
作用域门
一般来说,程序会在三个地方关闭前一个作用域,同时找开一个新的作用域。
它们分别是:class、module、def,它们被称为作用域门。class/module与def还有一点微妙的差别。在类或module中的代码会被立即执行。相反,方法中的代码只有在方法被调用的时候才执行。
扁平化作用域
my_var = "abc"
class MyClass
def my_method
end
end
怎么样在类MyClass中和方法my_method中使用my_var?
答案是这样的类无法使用my_var,只能改变类和方法的定义方式。下面来做这件事情:
my_var = "abc"
MyClass = Class.new do
puts my_var #这里可以使用了
define_method :my_method do
puts my_var #这里可以使用了
end
end
共享作用域:
假设想在一组方法之间共享一个变量,这个变量又不想被别的方法访问,就可以把这些方法定义在一个扁平作用域中:
def define_methods
shared = "abc"
Kernel.send :define_method, :my_method1 do
puts shared
end
Kernel.send :define_method, :my_method2 do
puts shared
end
end
instance_eval(): 打开接收者。
代码块
从底层看,使用块需要分两步:1、打包备用,2、调用块(yield)来执行代码。
在ruby中绝大多数都是对象,但是块不是。要存储一个块,你需要一个对象。Proc类可以做到。这个类提供了两个内核方法用于把块转化为对象:lambda()和proc(),还有两种方法也能做到:Proc.new()和&操作符。
lambda()和proc()大多数情况下可以通用。但有些区别:
1、在lambda中return仅仅从这个lambda中返回,在proc中return是从定义proc的作用域中返回。
2、参数数量:lambda中如果参数数量不对,会失败,抛出一个ArgumentError的错误。而proc会把传来的参数调整为自己期望的参数形式。