Ruby中的*eval方法

摘要

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}    # => #

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模块定义的私有方法,只能在对象内部调用,执行字符串中的代码,并返回结果。

你可能感兴趣的:(Ruby中的*eval方法)