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

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

系列文章原载于自己的博客,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

共收到 6 条回复
96

类 本身是类,但同时是 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

这样举例更明白一点呢。

3407e5

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

6829

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

8766

精辟

5楼 已删除
11124

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

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