Ruby中的类模型学习笔记

来源:互联网 发布:求救暗号 知乎 编辑:程序博客网 时间:2024/05/21 11:37

学习Ruby的初衷是由于ROR,虽然,学习ROR似乎不需要对Ruby语言有过多的了解。但是,在折腾ROR两个星期后,对其中的ActiveRecord发生了兴趣,ROR是怎么做到的?抱着这样的想法,尝试了解ROR背后的Ruby,粗略的看了一遍<<Programming Ruby>>,看完之后,感受最深的是:Block,迭代器,以及强大的正则表达式。但是,在着手做点什么的时候,发现还是很多的东西不懂,如method_missing是什么?attr_accessor如何实现的?而且隐约觉得,如果Ruby语言只是如我所了解的这些,那么,它的强大之处在哪里呢?以我现在的知识看来,它只是方便一点,却没有体现出强大,在网上搜索学习曲线,推荐一本叫做《Ruby元编程》的书,下载下来,读到第四章,这才认识到Ruby的强大之处。好吧。废话不说了,直接进入Ruby的类模型把。

1、Ruby中的类和其它静态编译语言的类有很大的不同,你甚至可以直接在类定义中编写代码,如:

class Test    puts "hello"end

所以从这一点来看,Ruby和其它语言(其它语言是指:静态编译形语言,如C++,Java等,以后简称这些语言为其它语言)就有很大的不同,你必须要转变一个思维:类定义不再是传统意义上的对对象行为的约束,而是在运行一段代码。实际上Ruby中的类都是开放的。所以对对象没有约束可言了。

2、类和模块本身也是对象。跟方法和块一样,类定义也会返回最后一条语句的值,如:

test = class Test    self                 #=>Test              end

在类定义时,类本身充当了当前对象self的角色。实际上类也有一个对象,所以定义类,是在操作一个实例而已。class定义的类名在Ruby中是一个常量,如果用以下方法定义一个类,也许可以更清楚的明白这种关系。

class Testdef sayHiputs "hello world!"endendTest.new.sayHiAnotherTest = Class.new dodef sayHiputs "hello world!"endendAnotherTest.new.sayHi

3、Eigenclass,在Eigenclass之前,先复习一下,对象实例的方法调用查找图谱,如下代码:

class Testdef sayHiputs "hello world!"endendt = Test.new

它的图谱如下:



当调用t.sayHi时,会向右一步,查找到t所属的类Test,在Test的实例方法中查找sayHi。如果查找到即返回,如果未找到,则继续在Test的superclass一级一级往上查找(当有某个类包含Module时,则Module的遍历顺序在当前类和其超类之间。如上图Kernel),直到查找到未知,如果整个下来都未查找到,则调用method_missing方法。这个方法确实会抛出一个异常,通知方法未找到。

Ruby支持给单个对象增加方法,即单件方法(singleton method),如下代码:

t = Test.newdef t.debugputs "debug"endt.sayHit.debug
但调用t.debug的时候,再按照以上图谱查找,你会发现无法找到debug方法,但是却没有抛出方法未找到的异常。那么debug方法究竟存放在哪个地方呢?首先对象实例只存储实例变量并不存储实例方法,所以排除。方法都存储在类中,那么debug方法是在t的类Test中嘛?或者是Test的超类Object中嘛?答案是否定的,如果存放在Test类中,那么Test类的所有实例均可以调用方法debug了。正确答案是,存放在Eigenclass,每个对象都有一个隐藏的Eigenclass,而这个对象的单例方法均存放在该处。Eigenclass是一个隐藏类,如下方法可以获取t的Eigenclass:

t_eigenclass = class << tselfend

当然,如果你在eigenclass域中定义方法。和def t.denug得到的结果是一致的,如下:

class << tdef debugputs "debug"endendt.debug

类实际上是Class类的一个实例,类名可以看成一个常量,所以类方法,也是单件方法。它同样保存在这个类的Eigenclass中,如下,两种方式定义的类方法是一致的:

class Testdef self.test; endendTest.testclass Testclass << selfdef test; endendendTest.test

在类的模型中加上Eigenclass,即得到了类模型和方法查找的全部了。








原创粉丝点击