Ruby中的类模型学习笔记

本文深入探讨了Ruby语言的类模型,介绍了Ruby类与静态编译语言中的类的区别,包括类的定义、类作为对象的概念、以及Eigenclass的使用等核心特性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

学习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 Test
	def sayHi
		puts "hello world!"
	end
end

Test.new.sayHi

AnotherTest = Class.new do
	def sayHi
		puts "hello world!"
	end
end

AnotherTest.new.sayHi

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

class Test
	def sayHi
		puts "hello world!"
	end
end

t = Test.new

它的图谱如下:



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

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

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

t_eigenclass = class << t
	self
end

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

class << t
	def debug
		puts "debug"
	end
end

t.debug

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

class Test
	def self.test; end
end
Test.test

class Test
	class << self
		def test; end
	end
end
Test.test

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








评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值