我手头上有一个 Java 项目,在过去几年的时间里,我小心翼翼地为之维护一个 Ant 构建脚本。Ant 能够执行大量任务,这一点我十分欣赏;然而,我常常发现,Ant 脚本的 XML 语法编写起来有些麻烦。而且,在 可表达性 方面,Ant 的 XML 部分还存在限制。实际上,当我发现自己需要更高程度的灵活性(例如在条件逻辑方面)时,我常常不得不在 Ant 的 script 任务中编写(例如使用 Groovy)定制任务或嵌入式逻辑。
告别表达乏力
关于本系列
作为开发人员,我们的工作就是为终端用户实现过程自动化;然而,很多开发人员却忽略了将自己的开发过程自动化的机会。为此,我编写了 让开发自动化 这个系列的文章,专门探讨软件开发过程自动化的实际应用,并教您何时 以及如何 成功地应用自动化。
数年前,当我在寻找可表达性更强的软件构建工具时,我听说过 Rake。Rake 是用于使用功能全面的 Ruby 编程语言构建软件的一种域特定语言。通过 Rake,我可以轻松获得正在寻找的可表达性。例如,我不必编写定制任务,因为 Rake 构建脚本是 Ruby(而不是 XML);因此,如果我需要某种条件逻辑,那么很容易在需要的地方编写这种逻辑。而且,Rake 采用基于依赖的任务系统,这类似于 Ant 及其前辈 make 中的任务系统。例如,如果输入 rake deploy,Rake 将调用之前设置的依赖任务,例如编译,运行测试等等。而且 Rake 非常智能,它只运行一次依赖性任务(就像 Ant 一样)。
对于大多数 Java 项目来说,Rake 本身并不能提供多大帮助;然而,一个相对较新的项目 Raven 将 Rake 的功能和可表达性与用于构建 Java 应用程序的特定任务结合起来。有了 Raven,Java 开发人员就有了一个真正可行的构建平台,该平台可提供特定于 Java 的特性,例如使用 javac 进行编译以及使用 jar 和 war 归档二进制文件,同时还拥有 Ruby 编程语言的丰富的可表达性。
什么是 DSL?
域特定语言(DSL)是专门为特定任务设计的编程语言。 Java 语言和 Ruby 是通用的编程语言,而 Ant 和 Rake 是专用于构建软件的 DSL。Rake 被认为是一种内部 DSL,因为它是用 Ruby 编写的,并且可以使用 Ruby 进行扩展。而 Ant 是一种外部 DSL,因为它首先被解析,然后在另一种语言中执行(也就是说,首先解析 XML,然后通过 Java 程序来运行)。
Raven 是 Rake,Rake 是 Ruby
在将 Raven 用于 Java 项目之前,理解 Raven 与 Rake 之间的关系会有所帮助。图 1 显示了 Raven 的一个逻辑图,可以看到,Raven 使用 Rake、RubyGems 和 Ruby。相应地,在使用 Raven 时,可以使用本地 Rake 任务。而且,还可以在 Raven 脚本中使用 Ruby。RubyGems 是类似于 CPAN、RPM 或 APT 的一个包管理器,用于下载必需的 Ruby 包(以及 Java jar)和任何依赖包。
图 1. Raven 的逻辑图