Oracle Java平台组首席架构师Mark Reinhold发表了一份关于模块化系统的情况的报告,强调了模块化的目标是什么。由于和已经存在的框架存在明显的重复,特别是OSGi,报告引起发了人们的讨论。
正如报告中所解释的,以及在JSR-376和模块化系统项目主页中完整的详细说明,模块化系统是为了解决当前Java访问模型中的两个疏漏:
可靠的配置:当前一个组件通过类路径访问另一个组件的类时相当容易出错,特别是尝试使用不在类路径里面或者存在多个版本的类的时候。
强封装:没有办法去限制特定组件暴露给其他组件的类,外部能访问所有的公共类。
完整的细节能够在报告和InfoQ之前的文章中找到,总的来说,每一个组件通常(但不一定)作为一个jar文件,其中包含了如下结构的模块描述文件module-info.java:
module com.foo.bar {
requires com.foo.baz;
exports com.foo.bar.alpha;
exports com.foo.bar.beta;
}
文件结构中包含一行或多行exports用于指出能够被其他组件访问的包,零行或多行requires用于指出自身需要访问的其他模块。该系统提供 了方法用于在编译时评估访问类型是否具有正确的可见性(例如声明为公共类并被所需组件导出),并在运行时评估必需的模块是否可用,而不需要去检查完整的类 路径。类似于OSGi中的清单文件。
OSGi的背景
OSGi是一个基于Java的模块化系统和服务化平台,实现了一个完整的动态的组件模型。从1998年在JSR-8第一次提出,在随后的审核中被延 迟发布(最近一次是2014年),OSGi定义了bundle(类似模块),采用包含如下的MANIFEST.MF文件的JAR文件的形式:
Bundle-Name: Hello World
Bundle-SymbolicName: org.wikipedia.helloworld
Bundle-Description: A Hello World bundle
Bundle-ManifestVersion: 2
Bundle-Version: 1.0.0
Bundle-Activator: org.wikipedia.Activator
Export-Package: org.wikipedia.helloworld;version="1.0.0"
Import-Package: org.osgi.framework;version="1.3.0"
(例子来源于Wikipedia)
显而易见的,尽管格式不一样,但是目的表现的和Java模块系统很相似。的确,Java平台模块系统和OSGi之间的相似性,在2005年提出的 JSR-277,“Java模块系统”中,初次尝试模块化Java时就已经被注意到了。最初瞄准的Java 7,JSR-277专注简化分发和执行Java程序包。尽管有和JSR-376几乎一样的名字,但两者最初的目标有着细微差别;虽然它的任务是修复“可靠 的配置”问题,但是它并不试图解决“强封装”的问题。相对于JSR-376,它也尝试增加一个版本模型到Java程序包中。这些目标和OSGi提供的功能 之间的相似性是完全足够让作者在最初考虑把OSGi作为一个解决方案,放弃的原因考虑为OSGi的版本控制太弱了。
不久之后创建的JSR-294的目标是实现“改进Java程序语言的模块性支持”。同时针对Java 7,JSR-294新增了模块概念(所谓的“超级包”)来修复强封装的问题;这个概念匹配了当前的Java平台模块系统项目。当Java 7的目标被放弃时,JSR-277和JSR-294从2012年开始被冻结,由JSR-376代替。
OSGi和模块化Java的另一个关联能在JSR-291中找到,“Java SE的动态组件支持”(本质上是OSGi服务平台发布的第四版)。JSR-291参考了JSR-277,作为最早的Java模块系统,来主动区分两者之间 的不同作用:JSR-277关注Java中使用的静态模块定义,而JSR-291关注运行时动态组件的加载和卸载。
最后,JSR-376也参考了OSGi,放弃它作为有效解决方案的主要原因是因为它的范围远远大于Java框架模块系统规范。
综上所述,多数人难以区分新的模块系统和OSGi是正常的。但是,模块系统和OSGi相互补充,用以服务于不同的目标,OSGi作为一个构建于 Java之上的框架,在一个持续运行的应用中,创建了一个用以动态管理bundle的环境,而模块系统作为Java本身的新能力能够更紧密更简单的控制静 态模块。
Java平台模块系统和OSGi的区别为了更好的理解这一点,InfoQ请教了Holly Cummins,《Enterprise OSGi in Action》的作者之一。不过以下内容并不会彻底说清楚Java平台模块系统和OSGi之间的差异,只是为读者提供一个基本的理解关于两者目标的区别。
一方面,Java新的模块系统将会提供更简单的方式在编译时检查包和类的可见性,但是当我们询问OSGi的bundle是否能用相同的方式使用时,Holly表示“答案是相当的复杂”。