引言
本文是本系列的第 1 部分,我们将开发包含客户端和服务器端组件的订单应用程序。然后将这些组件打包为 OSGi 包。客户端调用服务组件处理订单。服务组件具有处理订单和打印订单 ID 的方法。阅读本文后,您可以应用 Apache Felix 的概念和功能来将 Java 组件类构建和打包为 OSGi 包。
系统要求
要运行本文中的示例,请确保已在计算机上安装和设置了以下软件:
接下来,设置以下环境变量(按照示例 ANT_HOME=C:\apache-ant-1.7.0 进行设置):
接下来,将以下内容添加到 PATH 环境变量:
回页首
OSGi
OSGi 规范以更为动态的方式定义和传播 Java 应用程序的模块化。通常,Java 应用程序模块化为 JAR 包。但是使用 JAR 文件有局限性:
为了处理上述问题,可以使用 OSGi 框架,因为其中对 Java 的模块化系统进行了重新定义。相对于传统的 JAR 模块而言,基于 OSGi 的系统具有以下优势:
目前有三个已知的 OSGi 实现:
本系列文章将介绍如何将 Felix 作为 OSGi 容器使用。本系列文章涵盖以下主题:
回页首
订单应用程序
接下来让我们看看如何使用基于 Felix 的 OSGi 框架创建订单应用程序包。此应用程序包括两个组件:OrderClient.java(客户端)和 OrderService.java(服务器端)。客户端组件打包为 client.jar,服务器组件打包为 order.jar。接下来让我们首先看看 OrderClient
类。
public class OrderClient implements BundleActivator { private ServiceTracker orderTracker; private OrderService orderService; public void setService(OrderService orderService) { this.orderService = orderService; } public void removeService() { this.orderService = null; } public void start(BundleContext context) throws Exception { orderTracker = new ServiceTracker(context, OrderService.class.getName(), null); orderTracker.open(); OrderService order = (OrderService) orderTracker.getService(); if (order == null) { System.out.println("Order service not available"); } else { order.processOrder(); } } public void stop(BundleContext context) { System.out.println("Bundle stopped"); orderTracker.close(); } } |
正如清单 1 中所示,OrderClient
对 OrderService
组件调用 processOrder
方法,以在启动 client.jar 包时打印 orderID
。此类实现BundleActivator
接口,该接口具有两个回调方法:start()
和 stop()
。启动和停止客户端包时,Felix 容器将调用实现的 start()
和stop()
方法。
接下来让我们仔细分析一下 start()
方法。在 start()
方法中,首先获得 ServiceTracker
类的引用。您可以将此类视为工厂类。然后通过调用此 getService
方法从此工厂类获取服务引用。ServiceTracker
使用以下参数构造:包上下文和 OrderService
类的名称。
public class OrderServiceImpl implements OrderService, BundleActivator { private ServiceRegistration registration; public void start(BundleContext context) { registration = context.registerService(OrderService.class.getName(), this, null); System.out.println("Order Service registered"); } public void stop(BundleContext context) { System.out.println("Order Service stopped"); } public void processOrder() { System.out.println("Order id: ORD123") ; } } |
服务器端 order.jar 文件包含两个组件:OrderService
接口和 OrderServiceImpl
类。此接口具有由 OrderServiceImpl
类实现的抽象类 processOrder
。此类还实现了 BundleActivator
接口,此接口供 Felix 容器调用来启动和停止 order.jar 包。start()
方法在注册中心注册 OrderService
组件,以供客户端包使用。注册与导出对象供其他包使用的作用一样。
回页首
通过 Manifest 通信
真正的问题是,客户端包如何知道注册的服务包?此通信通过使用 Manifest 文件处理。在 Manifest 文件中,在导入和导出程序包中提供包的引用。客户端包 Manifest 通常导入服务组件程序包,而服务包 Manifest 导出自己的程序包。请注意,当包 A 导入包 B 的程序包时,包 B 必须导出自己的程序包。没有恰当的导入和导出定义,通信将失败。
Manifest-Version: 1.0 Bundle-ManifestVersion: 2 Bundle-Name: Order Service Client Bundle-SymbolicName: orderclient Bundle-Version: 1.0.0 Bundle-Activator: order.client.OrderClient Import-Package: org.osgi.framework, org.osgi.util.tracker, order |
Manifest-Version: 1.0 Bundle-ManifestVersion: 2 Bundle-Name: Order Service Bundle-SymbolicName: orderservice Bundle-Version: 1.0.0 Export-Package: order Bundle-Activator: order.impl.OrderServiceImpl Import-Package: org.osgi.framework |
对于订单应用程序,client.jar 包 Manifest 中包含条目 Import-Package: org.osgi.framework, org.osgi.util.tracker, order
。这个实际上表示,客户端包导入核心 OSGi 程序包和 OrderService
程序包。类似地,order.jar 包清单包含条目 Export-Package: order
。即,包导出其程序包供客户端使用。如果导入和导出未显式声明,OSGi 将引发运行时错误。
Manifest 文件还包含其他信息,如包激活器类的名称等。激活器类负责在包中调用 start()
和 stop()
方法。在本例中,client.jar 包的激活器类为 OrderClient
,order.jar 包的激活器类为 OrderService
。
回页首
部署
在部署和使用包前,请进行以下工作:
下一步便是部署这些包。请通过以下步骤,使用 Felix OSGi 容器部署客户端和服务包:
执行了上述构建文件后,将分别在 client/bin 和 service/bin 文件夹中创建 client.jar 和 order.jar 包。
每次启动 Felix 运行时,都会提示配置文件名称。配置文件名称的作用类似于项目名称。可以提供任何名称作为项目配置文件的名称。对于 Felix 运行时的每次后续启动,如果提供了之前输入的相同配置文件名称,Felix 将会加载与该项目名称关联的所有已安装包。如果提供了新配置名称,则将需要再次显式安装各个包:
install file:service/bin/order.jar
install file: client/bin/client.jar
start service_bundle_id
start client_bundle_id
为了指示包的 ID,可以使用 ps
命令。必须首先启动服务包,然后启动客户端包。在启动相应的包时,将显示以下输出(请参见图 2)。
还可以通过在 Felix Shell 提供 update bundle_id
命令更新包。包将在运行时动态更新。
回页首
结束语
本文简单介绍了 OSGi 框架的功能和概念,并说明了如何使用其创建动态 JAR 包。您了解了如何构建并将组件打包为 OSGi 包,然后在 Felix 运行时环境中运行。我们还了解了如何创建包 Manifest 文件,以作为包之间的通信接口使用。
OSGi 提供了一种新的构建和管理 JAR 包的方式。请继续关注本系列的第 2 部分,我们将介绍 Spring 框架在 OSGi 环境中如何代替 OSGi 承担管理包的责任。
资源使用说明:
下连接的资源中,felix版本为 1.4.0,所需要的环境文中已经说明。需要注意的是:
在运行ant编译service以及client时,修改client/build.xml 中的felix_home 变量,否则会出现编译出错。
使用到的下载地址:
http://download.csdn.net/detail/icecream0/4187317