随着 Java SE 对模块化功能原生支持的一再推迟(据最新的消息, Jigsaw项目—— Java SE中的 标准模块系统 —— 的开发工作相对滞后,已经延期到了 Java 9中 ,而 Java 9 到 2015 年才有希望发布), OSGi 作为事实标准的 Java 模块化实现得到了长足的发展,尤其是随着云计算和分布式技术的兴起, OSGi 这种模块化、动态部署的核心功能将会受到越来越多的重视。
目前 OSGi 规范的主流实现框架是 Eclipse Equinox 以及 Apache Felix ,其实际的应用场景也早就从起初的嵌入式设备扩展到桌面应用甚至 Java 企业级开发领域。最为大家所熟知的 OSGi 应用是 Eclipse IDE ,其实像 Sun GlassFish 、 IBM WebSphere Application Server 、 JBoss Application Server 以及 Apache 的大量开源项目如 Camel 、 Sling 、 ServiceMix 、 Kafaf 等都使用了 OSGi 技术,随着其应用的日益广泛,技术的可用性和成熟度经受了充分的考验,实践证明 OSGi 是一项可靠的技术,完全可以应用于企业级应用的开发。尤其值得一提的是,随着 OSGi Core Release 5 和 OSGi Enterprise Release 5 版本的发布,预计 OSGi 技术将会受到更多的关注,国内外已经有将 OSGi 应用于云计算和分布式计算的案例。
本文将选取 OSGi 应用于传统 Java Web 开发的场景,介绍借助于 OSGi 技术,怎样实现将 web 应用拆分成满足 OSGi 规范的 bundle ,实现组件的动态部署。目前,为了实现在 bundle 中支持 JSP 、 Servlet 等 Java Web 开发技术和规范,有两种主要的开发部署方式,本文都将进行介绍,但是这两种方案在实现上与传统的 Java Web 应用开发模式有较大的不同,且对 JEE 规范的支持尚不完整 。
按照Eclipse Equinox的习惯,会将bundle称为插件,在后文中会根据具体的应用场景穿插使用这两个名词,但其本质是相同的。
<!--[if !supportLists]-->一、 将Web容器作为bundle置于 Equinox中进行开发
在这种模式下,我们需要将支持 JEE 规范的容器也以 bundle 形式发布到 OSGi 容器中,而包含 JSP 、 Servlet 的 bundle 将会使用该容器接受对请求的访问,这种方式的好处在于 OSGi 容器管理整个运行环境,是个纯粹的可热插拔运行环境,其缺点在于需要 Web 容器支持以 bundel 的形式进行发布,对运行环境有一定的限制,目前支持该要求的 Web 容器有 Tomcat 和 Jetty ,但是主流的企业级应用服务器尚不支持这种部署方式。
鉴于 Eclipse 开发环境中已经包含了 Jetty 服务器,所以我们将会使用该服务器进行 Web 应用的开发。
首先我们需要使用 Eclipse 的工程向导创建一个插件工程。根据工程的需要,我们不生成启动类( activator ),不使用任何模版,插件过程如下:
新建一个插件工程:
此处我们不使用任何模版,因为我们不会用到 Eclipse 的 UI 功能:
为了测试进行 Web 开发,我们新建目录结构 /WebRoot/jsp (该目录名称是任意的,只需在后续定义扩展点时进行匹配即可)并在该目录下创建一个 JSP 文件,此时的工程结构如下:
为了使用 JSP 和 Servlet ,我们需要使用一个扩展点,该扩展的名称为 org.eclipse.equinox.http.registry.servlets ,我们需要使用该扩展点进行 JSP 目录和访问地址的匹配,在我们这个样例中我们匹配了 /WebRoot/jsp 目录,其访问地址的前缀将会是 /jsp/ :
除此以外,我们还需要添加对 org.eclipse.equinox.jsp.jasper.registry 插件的依赖,以实现 JSP 的编译功能。
最后,我们还需要在运行时配置里将 testweb 和 Jetty 所依赖的插件全部选中,如下:
然后点击“ Run ”按钮,此时就会启动 OSGi 运行环境,我们可以在控制台通过使用“ ss ”命令来查看运行插件的状态情况:
此时,我们可以在浏览器中输入 http://localhost/jsp/index.jsp ( Jetty 的默认端口是 80 ,可进行配置),就会看到我们所编写的 JSP 的代码了:
要实现访问 Servlet 和静态资源( CSS 、图片等),也需要类似的配置,下图是一个包含静态资源和 Servlet 的配置样例:
二、 将 Equinox 置于 web 容器中进行开发
以上的开发模式对于习惯于传统 JEE 开发模式的开发人员在接受上有一定的难度,直接使用 Equinox 作为运行环境在解决集群部署方面也有一定的难度,除此以外,主流的应用服务器并不提供 bundle 形式的 jar 以运行在 OSGi 容器中。所以,开发人员摸索出了另一种开发和部署模式。
在这种模式中,会将 Equinox 和我们开发的插件都打成一个标准的 war 包的格式,因此可以运行在任意支持 JEE 的容器中。 Equinox 的生命周期是通过 Servlet 来进行控制。这种模式被称为桥接模式。 Equinox 官方提供了该模式的实现,后来的社区贡献者基于此进行了功能的完善。
将下载后的代码导入到 Eclipse 中,工程目录如下所示:
从该目录可以看出,这是一个典型 JEE 工程,可以运行在任意的支持规范的 Web 容器或应用服务器中,查看 web.xml ,我们可以看到这样的配置:
我们可以看到这个 Servlet 拦截了所有的请求,由该 Servlet 代理所有的请求。
在部署阶段,可以把插件打成 jar 包,放到 /WEB-INF/eclipse/plugins 目录下即可。如将我们上例中的 testweb 导出为 jar 包后,置于指定目录下。然后在 Tomcat 中运行该应用。访问 http://localhost:8080/bridge/jsp/index.jsp 就可以看到我们编写的 JSP 页面了。
使用以上的模式进行开发,对原有的开发和部署模式冲击较小而且支持所有的应用服务器。
参考资料:
http://www.ibm.com/developerworks/cn/web/0907_osgiweb_liuqing/