摘要
instance_eval:运行时,代码块的接受者会变成self。(84页)
class_eval:在一个已存在类(模块)的上下文执行一个块。(107页)
eval:执行字符串中的代码,并返回结果。(141页)
参考书籍 《Ruby元编程》(第二版)
BasicObject # instance_eval
运行时,代码块的接受者为self,因此它可以访问接收者的私有方法和实例变量。我们把传递给instance_eval的代码称为上下文探针(Context Probe)。因为它就像是一个深入到对象中的代码片段,并可以对那个对象进行操作。
class Test
def a_method
return yield if block_given?
p self
'no block'
end
end
obj = Test.new
obj.a_method {p self} # => main
obj.instance_eval { p self} # => #<Test:0x007f7f63965700>
v = ["test"]
obj.instance_variables # => []
obj.instance_eval {@v = v}
obj.instance_variables # => [:@v]
obj.instance_eval {@v} # => ["test"]
instance_eval可以打破封装,在irb中可以快速查看对象的内部细节。
instance_eval还有一个双胞胎兄弟,BasicObject # instance_exec,允许对代码块传入参数
class KlassWithSecret
def initialize
@secret = 99
end
end
k = KlassWithSecret.new
k.instance_exec(5) {|x| @secret+x } #=> 104
使用instance_eval还可以用来定义单件方法(127页)
s1, s2 = "abc", "def"
s1.instance_eval do
def swoosh!; reverse; end
end
s1.swoosh! # => "cba"
s2.respond_to?(:swoosh!) # => false
Module # class_eval
它有一个别名module_eval。
class_eval会同时修改self和当前类(106页),instance_eval只修改self(实际上instance_eval可以修改当前单件类,见118页)。
通过修改当前类,class_eval实际上打开了当前类,就像class关键字做的一样,但是比class关键字更灵活。
它可以对任何代表类的变量使用class_eval,而class关键字只能使用常量。另外,class关键字会打开一个新的作用域,这样会丧失当前绑定(77、143页)的可见性,而class_eval方法则使用扁平作用域(81页),这意味着可以引用class_eval代码块外部作用域的变量。
def add_method_to(a_class)
a_class.class_eval do
def m; 'Hello!' ; end
end
end
add_method_to String
"abc".m
class_eval也有一个双胞胎兄弟,Module # class_exec,允许对代码块传入参数。
Kernel # eval
Kernel模块定义的私有方法,只能在对象内部调用,执行字符串中的代码,并返回结果。
网友评论