`
hz_chenwenbiao
  • 浏览: 1010115 次
  • 性别: Icon_minigender_1
  • 来自: 广州
社区版块
存档分类
最新评论

亲历基本OSGI实例,进入另番思维领域(转)----包括打包发布为可执行文件

    博客分类:
  • OSGI
阅读更多

软件的模块越来越插件化发展了,连硬件都处处热插拔,软件更当如此。记原来有个 JPF(Java Plugin Framework),也能实现动态插件化,但要是有个业界标准的东西一般来说会更好的。于是轮到 OSGI(Open Service Gateway Initiative) 登台,OSGI 出来也有好几个年头了,应用也轰轰烈烈的,比如 Eclipse 3 开始不再使用原来的插件体系,完全用 OSGI 搭建。WebSphere 6.1 也全面改用 OSGI;JBoss、WebLogic、Spring DM,甚至是 BMW 车的控制系统中都得到了很好的应用。

前面讲到可以用 OSGI 作为你的微内核,微内核的好处可以打个这样的比方:一台 Linux 服务器出故障了,应用程序坏了、某些服务不能访问等,但只要它还来连入网络,SSH 还是活的,我们就有办法进去修复它,想安装、卸载什么都行。

OSGI 也能让你动态的增减服务,或者说动态的加载卸载类等资源。OSGI 中的资源称作 Bundle,所以如果是基于 OSGI 的程序,能够在网上找到许多有用的 Bundle 直接在线插入到你的软件系统中。Bundle 其实就是一个 jar 文件,只是在 MANIFEST.MF 中有些特殊的定义,每个 Bundle 靠实现的 BundleActivator 的去控制 Bundle 的生命周期和发布、监听框架的事件,或者说与框架进行通信。

OSGI 是个规范,它的实现目前有 Equinox、Knopflerfish、Oscar、Felix 等,由于天天与我打交道的是 Eclipse 以及它背后的 Equinox,所以这里的示例程序就使用 Equinox 吧。

本例是基于 《OSGI实战》这一 PDF 文档稍加改造而成,比原例要简单,因为尽量避免了引入其他的 Bundle,不是 Web   应用,而是做成了一个窗口程序。

一样的,也是个用户登陆验证程序,程序运行中可以动态切换验证方式(数据库验证或 Ldap 验证),也可以插入新的验证方式,如文件验证方式。基于此,我们需要建立四个 Bundle,分别为:

1. UserValidatorBundle -- 定义了 Validator 接口方法,此例中还作为与用户交互的 Bundle
2. DbValidatorBundle -- 实现了 Validator 方法,用数据库进行验证的 Bundle
3. LdapValidatorBundle -- 实现了 Validator 方法,用 Ldap 进行验证的 Bundle
4. FileValidatorBundle -- 实现了 Validator 方法,用文件进行验证的 Bundle

定义接口的 Bundle 需要导出定义接口的 package,而实现接口定义的 Bundle 需要导入前面导出的接口 package。

OSGI 有点像 SPI(Service Provider Interface) 模式,上层有个接口定义,然后有多几可能的实现提供者,运行时可以选择哪种实现,例如 common-logging 与 log4j, jdk log 那样的关系,但是 SPI 不具有 OSGI 的动态性。同样,在 OSGI 的某个 Bundle 中也需要定义一个服务接口,没有接口规约就没法指导方法如何去调用。映射到 SPI 模式就是,UserValidatorBundle 为服务定义者,其他 Bundle 为服务提供者。

在 Eclipse 中直接能开发 OSGI 的 Bundle,并支持可视化操作与调试。因为 Eclipse 本身就是用的 Equinox,所以不需要单独下载 Equinox SDK。只要你电脑上有 Eclipse,马上就可以窥探一下 Equinox 的模样。

Eclipse 中菜单 Run -> Run Configurations 窗口中,左边 OSGi Framework 中,右键 new,然后 Deselect All,只勾选几个 Bundle,点击右下角的 Run 按钮就行啦。

这时候会在 Eclipse 的 Console 窗口中出现一个 osgi> 提示符,在其中可以输入各种命令,如 ss 显示所有的 Bundle,如图:


也可以安装卸载 Bundle、启用停止 Bundle 等等。当然实际做出来的 OSGI 程序不会去带那么一个 osgi 控制台的,osgi 控制台下的操作都可以在你的程序中完成,或者作为一个真正的后台管理界面,用户是不会觉察到的。之所以放个 osgi> 出来,是为了开发、测试方便。

我们现在要做的就是像 osgi> 控制台下用 ss 命令显示出来的那种 Bundle。本例中所用的 Eclipse 是 3.5.2 版,其他 3.x 的版本可能会有细微的差异,应该不会有影响的。


正式开始动手


一. 建立 Plug-in Project 工程 UserValidatorBundle


填入工程名称 UserValidatorBundle,This plug-in is targeted to run with: 指定为 an OSGI framework standard,建立一个标准的 OSGI 工程。


输入Bundle的相关元数据信息,这面这些信息会反应在 META-INF/MANIFEST.MF 文件中。

Plug-in ID指的是Bundle的唯一标识,在实际的项目中可以采用类似java的包名组织策略来保证标识的唯一性;
Plug-in Version指的是Bundle的版本;
Plug-in Name指的是Bundle的更具有意义的名称;
Plug-in Provider指的是Bundle的提供商;

还要创建一个关键的 Activator 类了,要好好考量一下包名称。图中显示说这个是 Options(可选的),而其实在我们的例子中,还是是很重的,用来启动界面或注册服务。

不需要 <Next> 选择模板的话,直接点接 <Finish> 按钮就行了。Eclipse 会进入到 Plug-in Development 视图,并用 Plug-in Manifest Editor 打开 MANIFEST.MF 文件。也为你生成了前面指定的 Activator.java 文件。如下图:



二. 对外提供用户验证接口包

    1) 建立一个 Validator 接口


2) 设置要导出的对外提供服务的 package

在 MANIFEST.MF 编辑器中,选择 Runtime 标签,Exported Packages 中点击 Add 按钮,在弹出的窗口中选择 Validator 接口所在的 package com.unmi.login.activator。


这步操作,其实也就是在 MANIFEST.MF 中添加了一行

Export-Package: com.unmi.login.service

好了,创建 UserValidatorBundle 的工作暂时就告一段落了。

三. 创建其他几个 Bundle

按照创建 UserValidatorBundle 相类似的方法,创建其他几个 Bundle,注意选择好每个 Bundle 的 Activator 实现类的位置。

1) 导入服务接口 package

另外创建好了 DbValidatorBundle、LdapValidatorBundle、FileValidatorBundle 之后,需要为它们导入 UserValidatorBundle 导出的 package,以明示自己是 Validator 接口的提供者。

操作方法是,各自的 MANIFEST.MF 编辑器的 Dependencies 标签页中,Imported Packages 里点 Add 按钮,弹出窗口中选择 com.unmi.login.service 包,其实就是为编程指定了依赖的包。


这步会在 MANEFEST.MF 中产生一行:

Import-Package: com.unmi.login.service

2) 编写实现 Validator 接口的实现类

因为是演示,所以验证过程硬编码,比如在 LdapValidatorBundle 中的 Validator 接口的实现类 com.unmi.login.service.impl/LdapValidatorImpl.java 代码为


其他几个 Bundle 中的 Validator 实现类的代码也是类似。

3) 实现 Bundle 的 BundleActivator 接口

前面讲过,Bundle 的 BundleActivator 实现是用来管理自身的生命周期和与框架交互的,所以虽要在各自的 BundleActivator 实现中,当启动 Bundle 时,注册自己以让框架能查找到该 Bundle, 停止 Bundle 时,把自己从框架中注销掉,以释放相关资源。先来编写那几个服务提供者的 BundleActivator 实现,如 LdapValidatorBundle 的类 com.unmi.login.activator.Activator.java 的代码如下:


注意其中的代码是怎么向 BundleContext 中注册自己的实例的,以及如何卸载,了解 BundleContext.registerService() 方法各参数的意义。

其他几个 Bundle,如 FileValidatorBundle 和 DbValidatorBundle 的 BundleActivator 实现代码也是类似,请仿照之。但是 UserValidatorBundle 的 Activator 类的实现就不太一样了,因为 UserValidatorBundle 是个服务定义者,而且这里也拿它来作为打开程序界面的入口,所以它的 Activator 实现的 start() 方法中就会安排安多做些事情的,也就单独拿出来说明。

4) 编写 UserValidatorBundle 的 Activator


在该 Activator 的 start() 方法中会弹出一个登陆窗口,让用户输入用户名和密码,然后登陆。停止该 Bundle 时会把登陆窗口关闭掉。注意,因为其他的 Bundle 不需要在上下文中查找本 Bundle 的实例来调用,所以在 start() 中没有向 BundleContext 注册自身的代码。

这里注意了,我们运行Bunble里,将选择要运行的bunble,被选择的bunbles都会从继承BundleActivator类的子类的start方法里执行,所以我们想要加在启动一个bunble时实现什么功能,我们只要写入到start方法里即可,如这里将写好的验证窗体放入到start方法里,那么一运行就会弹出验证窗体。

5) 调用服务的代码

完成在框架中查找可用的服务,如 Validator 实现类的实例,然后调用实现类的验证方法。这些代码实现在 LoginWindow 中,在 LoginWindow.java 中代码较多,完整代码可查阅文后附件,此处只列出关键性代码,即登陆按钮点击后所做的事情


到现在为止,所有的代码编写完毕,马上就可以在 Eclipse 中运行了。

四. 运行演示

演示之前,先对程序的行为进行简要的说明。程序运行后会跳出一个用户登陆窗口,在窗口中输入用户名和密码,点登陆按钮便能调用实际的验证实现类来验证用户,同时在窗口上可以看到登陆是成功,还是失败,以及失败时的原因为何,以登陆按钮旁边用个小字母标明当前是用的什么方式验证的。如图:


登陆按钮旁边用个字母标识了当前的验证方式,即框架中哪个 Bundle 在提供验证服务。N 代表未知,L 代表 LdapValidatorBundle,D 代表 DbValidatorBundle,F 代表 FileValidatorBundle。

1) 建立运行配置

Eclipse 的菜单 Run -> Run Configurations,OSGi Framework 上右键,new 一个 UserValidator 运行配置,在右边 Bundles 中选上我们创建的那四个 Bundle,可以分别设置它们的 Start Level 和 Auto-Start,例如这里设置 FileValidatorBundle 的 Auto-Start 为 false。在该页右下角可以直接点击运行。


启动后,出现 osgi 控制台,打印出 LdapValidatorBundle 和 DbValidatorBundle 启动的信息,并显示了登陆窗口。




现在就可以在 osgi> 控制台执行些操作来观察这个程序的运行状态,比如首先输入 ss 命令,看到加载了我们需要的 Bundle,而 FileValidatorBundle 是 RESOLVED,而不是 ACTIVE,那是因为我们建立运行配置的时候把该 Bundle 的 Auto-Start 设置成了 false。

osgi> 控制台下的基本操作有

ss 显示所有已加载的 Bundle,我们可以不时的执行这个指令,看看有哪些 Bundle 在
stop <id> 停止指定 id 的 Bundle,会触发该 Bundle 的 Activator 实现的 stop() 方法
start <id> 启动指定 id 的 Bundle,会触发该 Bundle 的 Activator 实现的 start() 方法
install <url> 安装一下 Bundle,指定 Bundle 的 jar 文件 url
uninstall <id> 卸载指定 id 的 Bundle,必须重新 install 后才能使用

还有很多命令,这里不详述,在 osgi> 提示符下乱输个它不认识的指令(如 dd),osgi> 便会提示出帮助来。

2) 多方演示

     (1) 直接登陆时的情形

               osgi> 控制台下先看下有哪些 Bundle 及状态,然后在登陆窗口中输入 Unmi/1234,登陆到:

登陆按钮旁边显示 L,控制台中显示 "使用 Ldap 进行登陆验证" 说明是用的 LdapValidatorBundle 验证的。

    (2) 停掉 LdapValidatorBundle 看看
           在 osgi> 控制台下执行 stop 1,再点击登陆窗口的登陆按钮,我们看到:


登陆按钮旁边显示 D,控制台中显示 "使用数据进行登陆验证" 说明是用的 DbValidatorBundle 验证的。

    (3) 把 DbValidatorBundle 也停了

          在 osgi> 控制台下执行 stop 3,再点击登陆窗口的登陆按钮,我们看到:


发现,现在没有活动的 Validator 实现 Bundle,所以无法登陆。

     (4) 把 FileValidatorBundle 启起来

        在 osgi> 控制台下执行 start 4,这时候把密码多加个 5, 再点击登陆窗口的登陆按钮,我们看到:


没问题,可以使用刚启用 FileValidatorBundle 来进行登陆验证了。

你还可以再次把沉睡的 LdapValidatorBundle 或 DbValidatorBundle 启动起来,或者干脆把某个 Bundle uninstall 掉,用 ss 都看不到了,要用的话必须 install 回来。

五. 发布基于 OSGI 的系统

前面都是在 Eclipse 环境中运行的基于 OSGI 程序,那我们离开 Eclipse 该怎么运行呢?我们也没有看到所谓 Bundle 即是一个 jar 那样的文件,写好的程序,最后的事情就是发布了。

1) 导出 Bundle 项目为 jar 包

Eclipse 菜单 File -> Exports,弹出窗口中选择 Deployable plug-ins and fragments



选择那四个 Bundle,设置输出文件的目录为 E:\workspace\OsgiDemo,然后点 Finish 按钮



完成后,在 E:\workspace\OsgiDemo 文件夹中,生成了一个 plugins 目录,里面就是刚刚导出的那几个 Bundle。


DbValidatorBundle_1.0.0.201003301134.jar
FileValidatorBundle_1.0.0.201003301134.jar
LdapValidatorBundle_1.0.0.201003301134.jar
UserValidatorBundle_1.0.0.201003301134.jar

2) 拷入所需的支持包

从 Eclipse 的 plugins 目录中拷贝 org.eclipse.osgi_3.5.2.R35x_v20100126.jar 到 E:\workspace\OsgiDemo 目录中。

3) 配置 config.ini

先要在 E:\workspace\OsgiDemo 目录中创建子目录 configuration,然后在其中建立文件 config.ini,内容为(注意osgi.bundles的值是按按我们实际打包出去的bunble名字来改它的值):

osgi.noShutdown=true

#避免Unable to acquire application service. Ensure that the org.eclipse.core.runtime错误
eclipse.ignoreApp=true

#因为使用了 swing,无该属性则报 java.lang.NoClassDefFoundError: javax/swing/JFrame
org.osgi.framework.bootdelegation=*

#这里有意没有加载 FileValidatorBundle,待以后 install
osgi.bundles=plugins/DbValidatorBundle_1.0.0.201003301134.jar@start,\
             plugins/LdapValidatorBundle_1.0.0.201003301134.jar@start,\
             plugins/UserValidatorBundle_1.0.0.201003301134.jar@start

osgi.bundles.defaultStartLevel=4


4) 建立批处理 run.bat

在 E:\workspace\OsgiDemo 中创建批处理文件 run.bat,内容为(这里也要注意一点就是你的org.eclipse.osgi版本不同下面那个版本时,你需要改为你的实际版本):

@echo off
java -jar org.eclipse.osgi_3.5.2.R35x_v20100126.jar -console

E:\workspace\OsgiDemo 中的文件目录结构如下:

E:\WORKSPACE\OSGIDEMO
│   org.eclipse.osgi_3.5.2.R35x_v20100126.jar
│   run.bat

├─configuration
│       config.ini

└─plugins
           DbValidatorBundle_1.0.0.201003301134.jar
            FileValidatorBundle_1.0.0.201003301134.jar
            LdapValidatorBundle_1.0.0.201003301134.jar
           UserValidatorBundle_1.0.0.201003301134.jar

5) 运行 run.bat

双击 run.bat 即可,这时候也出现 osgi> 控制台和一个登陆窗口,不过这时候是在 dos 窗口中的。你可以不让程序出现 dos 窗口,org.eclipse.osgi_3.5.2.R35x_v20100126.jar 是个可执行的 jar,又击它正常情况下就只会显示一个登陆窗口。你也可以去下载个 Equinox 程序的 Launcher 来启动 OSGI 程序。不过我们这里需 osgi> 控制台,所以要保留那个 dos 窗口,如下图:


刚启动时我们没有加载 FileValidatorBundle,我们执行下列命令

osgi>install reference:file:plugins/FileValidatorBundle_1.0.0.201003301134   #加载 FileValidatorBundle
osgi>stop 1 2 #停掉 DbValidatorBundle 和 LdapValidatorBundle
osgi>start 4     #启动 FileValidatorBundle

然后输入用户名和密码试下,从控制台输出和登陆按钮旁边的 F 标志,确实是使用了新安装的 FileValidatorBundle。

用 install <url> 你可以随时安装任何地方的 Bundle jar 包。

执行 stop 1 2 只是停掉 DbValidatorBundle 和 LdapValidatorBundle,并没有从 JVM 中卸载掉,用 ss 命令可以看到它们处于 RESOLVED 状态。这时候你也可以试着去 E:\workspace\OsgiDemo\plugins 中删除那两个文件 DbValidatorBundle_1.0.0.201003301134.jar 和 LdapValidatorBundle_1.0.0.201003301134.jar,系统会提示正在使用无法删除。这时候再执行下卸载命令

osgi>uninstall 1 2

再去删除那两个文是可以成功的,说明 JVM 真的卸载掉了这两个包。可以看出,OSGI 在通过 install 和 uninstall 动态的安装和卸载 Bundle 是多么的方便。

比如说原有系统中要添加一个功能模块,只要动态安装到框架中就能即使用了。原有系统中有个模块有 Bug,修改好,卸载再重装一下就 OK 啦,基本上系统类似于这种变动都不需要停掉服务,重启应用。

还是前面那句话,实际应用中是一般不会有这种 osgi> 控制台的,你想要在 osgi> 控制台下来做那些动人的操作也行。因为 Equinox 是纯 java 实现的,所以能在 osgi> 控制台下执行的命令,一定可以在你的程序里进行控制,或者以一种对于后台人员更友好的管理方式。

上面的例子只是引领你对 OSGI 有个基本的印象,进而真正领悟到 OSGI 的先进思想,这样才能在你的实际工作运用自如。当然我们在项目中不能因为技术而技术,如果是小项目我想大可不必把 OSGI 揉进来。具有一定规模的项目可考虑用 OSGI 框架去站在很高的层次去做个微内核,并且要合理的进行组件规划,才能灵活应动组件的动态插拔。

分享到:
评论
2 楼 harbey 2012-02-12  
部署显得好麻烦。如果是其他的环境呢,例如linux等又该如何部署呢?
1 楼 harbey 2012-02-12  
  很好!

相关推荐

    【转】亲历基本 OSGI 实例,进入另番思维领域

    标题中的“【转】亲历基本 OSGI 实例,进入另番思维领域”提示我们,这是一篇关于OSGi(Open Services Gateway Initiative)基础实例的分享,作者通过实践体验,引导读者理解OSGi这种模块化系统的新思维方式。...

    osgi资料打包

    OSGi(Open Services Gateway Initiative)是一种Java模块化系统和服务平台,它允许开发人员将应用程序分解为可独立更新和管理的模块。这个平台的核心是它的模块系统,也称为OSGi框架,它提供了一种动态、模块化的...

    osgi-resource-locator-1.0.1-API文档-中文版.zip

    赠送jar包:osgi-resource-locator-1.0.1.jar; 赠送原API文档:osgi-resource-locator-1.0.1-javadoc.jar; 赠送源代码:osgi-resource-locator-1.0.1-sources.jar; 赠送Maven依赖信息文件:osgi-resource-locator...

    spring-osgi-1.2.0-rc1-with-dependencies.zip

    OSGi的核心是其模块化系统,它将Java应用分解为独立的、可重用的模块,每个模块称为一个bundle。这些bundle之间通过服务进行交互,具有动态加载和卸载的特性,极大地增强了软件的灵活性和可维护性。 2. **Spring ...

    osgi-resource-locator-1.0.1-API文档-中英对照版.zip

    赠送jar包:osgi-resource-locator-1.0.1.jar; 赠送原API文档:osgi-resource-locator-1.0.1-javadoc.jar; 赠送源代码:osgi-resource-locator-1.0.1-sources.jar; 赠送Maven依赖信息文件:osgi-resource-locator...

    基于java的开发源码-OSGi 分布式通讯组件 R-OSGi.zip

    基于java的开发源码-OSGi 分布式通讯组件 R-OSGi.zip 基于java的开发源码-OSGi 分布式通讯组件 R-OSGi.zip 基于java的开发源码-OSGi 分布式通讯组件 R-OSGi.zip 基于java的开发源码-OSGi 分布式通讯组件 R-OSGi.zip ...

    spring-osgi-1.2.1-with-dependencies

    OSGi是Java平台上的一种模块化系统,它允许开发者将大型应用程序分解为独立的、可交互的组件,这些组件可以在运行时动态地安装、卸载和更新,提高了软件的可维护性和可扩展性。Spring OSGi则是Spring框架与OSGi规范...

    可运行的OSGI实例

    这个"可运行的OSGI实例"提供了实践OSGI技术的源码,确保用户可以直接运行并学习。 标题中的“可运行的OSGI实例”意味着这是一个已经配置好并经过测试的OSGI项目,用户可以下载并直接在本地环境中启动,体验OSGI的...

    carrot-osgi-anno-scr-make-2.0.1.zip

    在软件开发领域,OSGi(Open Services Gateway Initiative)是一种模块化系统和Java应用程序框架,它使得开发者能够创建可独立更新和重用的模块。在这个框架下,组件描述符(Service Component Runtime, SCR)是用于...

    killbill-osgi-bundles-lib-slf4j-osgi-0.8.4.zip

    【标题】"killbill-osgi-bundles-lib-slf4j-osgi-0.8.4.zip" 是一个基于OSGi的 Kill Bill 库,其中包含了SLF4J(Simple Logging Facade for Java)的OSGi兼容版本。SLF4J是一个为各种日志框架提供简单抽象的接口,...

    spring-osgi-1.2.1-with-dependencies.zip

    spring-osgi-1.2.1-with-dependencies.zip spring-osgi-1.2.1-with-dependencies.zip spring-osgi-1.2.1-with-dependencies.zip

    OSGi_with_Eclipse_Equinox_-_Tutorial

    OSGi的关键特性可概括为以下几点: 1. **模块化**:OSGi支持将应用程序分解为更小、更易管理的模块(称为“bundles”),这有助于提高代码的复用性和可维护性。 2. **运行时动态性**:OSGi框架能够在运行时加载、...

    OSGI 经典实例,入门

    本教程以经典实例为基础,旨在帮助新手快速入门OSGI,同时也对经验丰富的开发者具有参考价值。 首先,理解OSGI的基本概念至关重要。OSGI框架由一组称为"bundle"的模块组成,每个bundle都是一个自包含的Java程序单元...

    OSGI 实例eclipse插件开发

    OSGI(Open Services Gateway Initiative)是一种模块化系统和Java服务框架,它允许应用程序由一系列可独立更新和替换的模块组成,这些模块称为“bundle”。在本实例中,我们将探讨如何利用OSGI技术来开发Eclipse...

    Felix_OSGi实作

    使用Maven原型创建项目后,开发者可以进入项目目录,执行编译和打包命令来构建项目,从而创建出可部署的Bundle。 总的来说,OSGi与Felix的结合为Java开发者提供了一种强大而灵活的方式,以模块化的方式进行应用开发...

    OSGi HelloWorld实例

    OSGi(Open Services Gateway Initiative)框架是一个用于Java应用程序模块化的开放标准,它允许开发者将复杂的系统分解为小的、独立的模块,称为“服务”。在OSGi环境中,每个模块都有自己的类加载器,使得模块之间...

    maven-osgi-plugin-launcher-framework-equinox-1.0.15.jar

    maven-osgi-plugin-launcher-framework-equinox-1.0.15.jar

    atlassian-plugins-osgi-2.6.0.jar.zip

    《Atlassian Plugins OSGi 2.6.0.jar - 深入理解与应用》 Atlassian Plugins OSGi 2.6.0.jar 是一个重要的组件,它结合了Atlassian的产品生态与OSGi(Open Service Gateway Initiative)框架的优势。OSGi是一种模块...

    spring-osgi-1.2.0-with-dependencies.zip

    spring-osgi-1.2.0-with-dependencies.zip spring-osgi-1.2.0-with-dependencies.zip spring-osgi-1.2.0-with-dependencies.zip

Global site tag (gtag.js) - Google Analytics