分享 初学者连载系列之二十一:instance_eval 与 class_eval 的区别

kevinhua · 2012年04月07日 · 最后由 moliliang 回复于 2016年09月13日 · 14994 次阅读

系列文章原载于自己的博客,TOPI.CO (http://topi.co) ,某天不小心就 push 错啦,懒得从头再来,上传到 Ruby-China 来,一是方便自己回顾,另外也方便跟我一样的初学者

最近在看《Metaprogramming Ruby》的时候,看到代码块一章,在网上看到一篇介绍 instance_eval 与 class_eval 的文章:

###instance_eval

首先从名字可以得到的信息是,instance_eval的调用者 receiver 必须是一个实例 instance,而在 instance_eval block 的内部,self 即为 receiver 实例本身。

obj_instance.instance_eval do
  self  # => obj_instance
  # current class => obj_instance's singleton class
end

根据这个定义,如果在一个实例上调用了 instance_eval,就可以在其中定义该实例的单态函数 singleton_method

class A
end

a = A.new
a.instance_eval do
  self  # => a
  # current class => a's singleton class
  def method1
    puts 'this is a singleton method of instance a'
  end
end

a.method1
#=> this is a singleton method of instance a

b = A.new
b.method1
#=>NoMethodError: undefined method `method1' for #<A:0x10043ff70>

同样,因为类 class 本身也是 Class 类的一个实例,instance_eval 也可以用在类上,这个时候就可以在其中定义该类的 singleton_method,即为该类的类函数。

换句话说,可以用 instance_eval 来定义类函数 class method,这比较容易混淆,需要搞清楚。

class A
end

A.instance_eval do
  self  # => A
  # current class => A's singleton class
  def method1
    puts 'this is a singleton method of class A'
  end
end

A.method1
#=> this is a singleton method of class A
class_eval

###class_eval

再来看class_eval,首先从名字可以得到的信息是,class_eval 的调用者 receiver 必须是一个,而在 class_eval block 的内部,self 即为 receiver 类本身。

class A
end

A.class_eval do
  self  # => A
  # current class => A
end

根据这个定义,如果在一个类上调用了 class_eval,就可以在其中定义该类的实例函数 instance_method

class A
end

a = A.new
a.method1
#=> NoMethodError: undefined method `method1' for #<A:0x10043ff70>

A.class_eval do
  self  # => A
  # current class => A
  def method1
    puts 'this is a instance method of class A'
  end
end

a.method1
#=> this is a instance method of class A

换句话说,可以用 class_eval 来定义实例函数 instance method,这也比较容易混淆,需要搞清楚。

###总结

instance_eval 必须由 instance 来调用,可以用来定义单态函数 singleton_methods

class_eval 必须是由 class 来调用,可以用来定义实例函数 instance_methods

类 本身是类,但同时是 Class 的实例,(这个比较绕)。 所以我们用 A.instance_eval 和 A.class_eval 来定义 @xx 是一样的,但定义 def xx 就不一样了,前者是单键方法(类方法也是单键方法),后者是实例方法。

a = "a" a.instance_eval {def a; p 1; end} a.a # => 1 a.respond_to?(:class_eval) # => false

这样举例更明白一点呢。

今天复习了一下 Metaprogramming Ruby,觉得 instance_eval 还有一个很大的用处是用来做 上下文探针 ,比如动态的在某个对象中加入某个特定的属性。

#1 楼 @vkill #2 楼 @neverlandxy_naix 感觉有些时候元编程这本书讲的不透测,就像 instance_eval 和 class_eval,他没有说也没有解释为什么 instance_eval 的当前类就是单件类

5 楼 已删除

@kevinhua Ruby Metaprogramming 是一本深入浅出的好书!

需要 登录 后方可回复, 如果你还没有账号请 注册新账号