ActiveHibernate:使用 JRuby 实现 ORM

  • Werner Schuster
  • Jason lai

2007 年 8 月 17 日

话题:JavaRubyRuby on Rails语言 & 开发

对于面向对象编程(OOP)语言来说,对象关系映射(Object Relational Mapping,ORM)是一个内容甚广的话题。一旦出现了需要持久化的数据,我们就不得不马上面对如何存储这些数据的现实问题。在 Java 世界中人气最高的一项 ORM 解决方案,目前已经在 JRuby 项目中得到了支持。JRuby 团队成员 Ola Bini 在他一篇博客文章中为这个计划打响了第一枪——ActiveHibernate项目的诞生就是为了让 JRuby 开发人员也能用上Hibernate。我们采访了 ActiveHibernate 项目的实现人Johan Andries,他是一名在 AE 供职的咨询师。

在 Java 领域,Hibernate 一直是大红大紫的,在.NET 平台下,它还有一个堂兄弟 NHibernate。Johan 对 ActiveHibernate 的典型用户进行了阐释:

有些开发人员希望能充分利用 Rails 的生产力和其它优点,但他们已经在 Java 领域对象上投入了很多,或者出于某些原因需要比 ActiveRecord 更强大的东西。对于这些开发人员,我们是非常重视的。原因可能出现在以下几个例子中:组合关键字(Composite Keys)、遗留数据库、Prepared Statements、复杂的类层级映射(Complex Class Hierarchy Mapping)或者 Hibernate 的第一和第二级缓存。要使这一切成为现实,我们就得提供一个遵循 ActiveRecord 模式的接口,并且还能充分榨取 Hibernate 的强大潜能。

我们在一开始就把焦点放在对 Ruby 对象的映射之上。Charles Nutter 建议说,我们也应当重视已有的 Java 类。在最简单的情况下,你可以编写只带属性访问器的纯粹 Ruby 类,然后手动创建出常见的 Hibernate 映射文件。在我们的 Subversion 代码库中已经有了一些测试用例,就是这种情况的典型例子。直接处理 XML 映射文件,并不是 Ruby 的风格,因此我们也在着手尝试 DSL 的方式。

随后,我们的主题就被引到如何使用 ActiveHibernate 定义映射的问题上来。Ruby 由自身内嵌的领域特定语言(Domain Specific Language,DSL)受益良多,它的内嵌 DSL 使人们可以写出看起来更像一门针对领域问题的语言的 Ruby 代码,而不是更为通用的代码。(如果您想更多的了解 DSL,请在 InfoQ 上观看 Obie Fernandez 关于 DSL 的视频演讲)。

在 ActiveHibernate 的情况中,我们使用一个 DSL 用来配置从 Ruby 或者 Java 类型到关系型数据表中条目的映射,是非常有用的。在 Java 中,这是通过 XML 文件来实现的。Johan 解释了在 Ruby 中这一切看起来会是如何:

我们可以(而且也打算)搞出两个类似 DSL 的方式。其中一种形式是把映射信息嵌入 Ruby 类定义当中。Ola Bini(还有谁来着?)给我发了一份补丁,里面包含一些(元)编程技巧,使得我可以把代码写成下面这样:
class Project

include Hibernate

table_name = "PROJECTS" #optional

primary_key_accessor :id, :long

# column names are optional

hattr_accessor :date, :timestamp, :START_DATE

hattr_accessor :name, :string

hattr_accessor :complexity, :double

hattr_accessor :size, :long

hattr_accessor :on_schedule, :boolean

end
通过包含 Hibernate 模块(而不是从框架中的基类中继承),Hibernate 映射就会被自动生成并配置。工具类方法(如“save”和“find”和用于已映射属性的访问器方法也会随之被加入。值得注意的是,这是第一个可以正常工作的草案,我们仍然欢迎大家的建议。这部分代码很快就会出现在代码库中。

另一种形式则将映射信息和 Ruby(或者 Java)类分离开来,以确保它们完全可以不用考虑持久化细节。不过目前在这方面还没有开展什么工作。

Johan 接着解释了将 ActiveHibernate 移植到.NET、IronRubyNHibernate的移植版中将会有些什么:

IronRuby/NHibernate 的组合看起来会是最自然的候选方案。尽管 IronRuby 目前还处于 pre-alpha 阶段,但从我所看见的事实来说,很可能我们要做一些在 ActiveHibernate 上为 JRuby Java 扩展所做的相同的事情。不过 NHibernate 的部分会更为困难,NHibernate 基于的是 Hibernate 2,但引入了不少 Hibernate 3 的优秀特性。不巧的是,ActiveHibernate 采用了很多 Hibernate 3 的新特性(比如在 Hibernate 3 中加入,用于支持动态映射表和 XML 序列化的 Tuplizer),这些特性还没有被移植到 NHibernate。因此,一切都和 NHibernate 的路线图息息相关。不管怎么说,还是很有可能将所有构成 ActiveHibernate 的 Ruby 代码拿到 NHibernate 上的(如果我们不做移植的话)。

最后,Johan 详细阐述了 ActiveHibernate 项目的计划:

ActiveHibernate 是我下班回家后的家庭作业,但我认为我们在一两个月内就应该会出真正可用完整的成果。当然,如果有其他有想法的人加入,并给我们带来补丁(就像 Ola 所做的),项目的进展会变得更快。大家现在就可以到http://code.google.com/p/activehibernate检出我们的代码,在那里你可以找到一份自述文件以及一个 Rakefile,这样你就可以不比大费周章地进行构建并运行测试了。(不过要记住的是,现在这些东西还问世不久,不是特别成熟。)

此外,Johan 也在撰写博客文章,记述 ActiveHibernate 的开发进度

查看英文原文:ORM with JRuby - ActiveHibernate

JavaRubyRuby on Rails语言 & 开发