2.1.3  HelloWorld(1)

在这一节中,我们将完成一个HelloWorld的例子。之前大家应该看到过Java语言、C语言或C++等语言中的HelloWorld,那些HelloWorld程序都是在运行后输出一个"Hello World",然后就结束了程序。我们今天的HelloWorld例子程序主要的功能是在启动和停止一个Bundle的时候来做些事情(输出信息)。下面我们就一步一步完成第一个Bundle。

第一步,创建Bundle工程。

在Eclipse中创建一个Plug-in工程(见图2-7)。

(点击查看大图)图2-7  创建插件工程

输入工程相关的信息,这里和创建普通的Java工程唯一的不同点就是要选择this plug-in is targeted to run with,在这里选择 an OSGi framework的standard选项,也就是说建立一个标准的OSGi Bundle工程(见图2-8)。

输入Bundle的相关元数据信息(见图2-9)。

Plug-in ID 指的是 Bundle 的唯一标识,在实际的项目中可以采用类似 java的包名组织策略来保证标识的唯一性;

Plug-in Version 指的是 Bundle 的版本;

Plug-in Name 指的是 Bundle 的更具有意义的名称;

Plug-in Provider指的是 Bundle 的提供商;

Execution Environment指的是Bundle运行需要的环境;

剩下的最关键的就是 Activator部分了,这里填入自己的一个类名就可以了,在工程建立时 Eclipse 会自动建立这个类。

(点击查看大图)图2-8  插件工程设置

(点击查看大图)图2-9  插件信息设置完成了Bundle工程的创建后,在Package视图中就可以看到如图2-10这样的视图,表明工程创建成功了。

图2-10  插件工程内容

第二步,完成Activator的代码。

打开默认的HelloWorldActivator.java代码(见图2-11)。

(点击查看大图)图2-11  默认Activator代码

2.1.3  HelloWorld(2)

可以看到HelloWorldActivator实现了BundleActivator接口,然后HelloWorldActivator中有两个空的方法--start和stop。其中,start方法是在Bundle被启动的时候调用的,stop是在Bundle被停止的时候调用的,下面我们在这两个方法中加入代码(见图2-12)。

(点击查看大图)图2-12  修改后的Activator代码

HelloWorld的例子已经完成了,下面看这个例子的运行效果。

第三步,运行。

首先创建一个HelloWorld用的运行配置(见图2-13)。

(点击查看大图)图2-13  运行配置的Bundles

然后我们设置HelloWorld的Auto-Start属性为false,并且将Target Platform中不需要的Bundle去掉。

再直接点击HelloWorld这个Bundle配置行后面的Auto-Start列的default,在下拉框中选择false,并点击Target Platform前的钩,使得Target Platform下面的项都变为不选择状态。接着我们点击右侧的"Add Required Bundles",这个时候发现Target Platform前的方框变为部分选择的状态,最后我们选中"Only show selected bundles"。这样就完成了运行配置的设置(见图2-14)。

(点击查看大图)图2-14  运行配置的Bundles

最后点击"Run",在Console中出现"osgi>"提示。这表明已经成功启动了我们的第一个OSGi应用。

在osgi>提示符下输入ss,然后回车,我们可以看到如图2-15所示的显示。

(点击查看大图)图2-15  运行结果可以看到,HelloWorld Bundle已经被安装并且完成了解析,但是还没有启动。下面我们在osgi> 提示符下输入 start 1,然后回车,看看会发生什么。这个时候再输入ss。应该有如图2-16所示的显示。

(点击查看大图)图2-16  启动Bundle后的结果

我们可以看到输出了"HelloWorldBundle started!",并且通过ss命令,看到HelloWorld Bundle的状态从前面的RESOVLED变成了ACTIVE,说明我们的HelloWorld Bundle已经成功启动了。并且在Activator的start方法中加入的输出信息也正确打印出来了。

接着,我们输入stop 1,并且再用ss命令查看Bundle的状态,会得到如图2-17所示的反馈。

(点击查看大图)图2-17  停止Bundle后的结果

这个时候HelloWorld Bundle已经被停止。我们在Activator中加入的输出信息正确地输出在了Console,并且通过ss命令看到HelloWorld Bundle的状态从刚才的ACTIVE变为了RESOLVED。

到这里我们完成了第一个OSGi Bundle,也尝试运行了第一个OSGi的程序。在下面的章节,将会看到更加复杂一些的例子。

2.1.5  从外部启动Equinox

前面,我们都是通过Eclipse启动我们的Bundle。但是,在有些时候,我们希望自己来控制OSGi的容器的启动,并且在OSGi的容器外部获取OSGi的服务,甚至是把OSGi的容器内嵌到我们的应用之中。下面我们就来看一下如何把Equinox嵌入到应用中。由应用来启动Equinox、获取OSGi的服务,以及加载OSGi容器中的其他插件的类。并且也会演示OSGi容器中的插件如何加载OSGi容器外的类的方法。

我们在前面演示了如何通过命令行来启动Equinox,常见的一种脚本为:java-jar plugins/org.eclipse.osgi_3.4.3.R34x_v20081215-1030.jar -configuration configuration -console,然后在当前的configuration目录下放置一个config.ini,在此config.ini中通过 osgi.bundles=来配置要加载和启动的插件,例如osgi.bundles=example.jar@start,那么要在程序中启动Equinox容 器,其实是做差不多的事情。

通过查看Equinox的代码,会看到调用上面的org.eclipse.osgi.jar后执行的是EclipeStarter中的静态run方法,因此只须在外部传入合适的参数,并调用此run方法即可完成Equinox的启动,在程序中启动Equinox,通常希望做到的是能够指定config.ini 的配置信息及插件的位置,而不是由Equinox去决定。如果不进行设置,默认情况下EclipseStarter将会在工作路径下产生 configuration,并以该configuration目录下的config.ini作为Equinox启动的配置。对于 osgi.bundles配置的bundle的路径,默认则为当前EclipseStarter代码所在的目录,例如上面的命令行,Equinox在启动时就会从plugins目录中去加载插件,这通常是无法满足在程序中启动Equinox的需求的。如果想自定义Equinox启动的配置信息,而不是通过加载指定的configuration中的config.ini,那么可以在程序中调用 FrameworkProperties.setProperty来设置启动Equinox的配置信息。如希望指定osgi.bundles中指定加载的bundle的相对路径,那么可以在Equinox启动的配置信息中增加osgi.syspath的设定:FrameworkProperties.setProperty("osgi.syspath",你希望指定的bundle所在的路径)。Equinox启动的配置信息还有很多种,有具体需要的话可以查看EclipseStarter中processCommandLine的方法。通过这样的方式,就可以启动Equinox:EclipseStarter.run(new String[]{"-console"},null);按照上面的方式就可以实现在外部程序中启动equinox了。

OSGi通过BundleContext来获取OSGi服务,因此想在OSGi容器外获取OSGi服务,首要的问题就是要在OSGi容器外获取到BundleContext,EclipseStarter中提供了一个getSystemBundle- Context的方法,通过这个方法可以轻松拿到BundleContext,而通过BundleContext则可以轻易拿到OSGi服务的实例。不过这个时候要注意的是,如果想执行这个OSGi 服务实例的方法,还是不太好做的,因为容器外的classloader和OSGi服务实例的class所在的classloader并不相同,因此不太好按照java对象的方式直接去调用,更靠谱的是通过反射去调用。

如果想在容器外获取到OSGi容器里插件的class,一个可选的做法是通过BundleContext获取到Bundle,然后通过Bundle来加载 class,采用这样的方法加载的class就可以保证是相同的。否则会出现容器外的一个A.class不等于容器里插件的A.class,其中原因对于稍微知道java classloader机制的人都是理解的。

按照上面的说法,一个简单的启动Equinox及与OSGi容器交互的类可以这么写:/**

* 启动并运行equinox容器

*/

publicstaticvoidstart()throwsException{

// 根据要加载的bundle组装出类似a.jar@start,b.jar@3:start这样格式的

osgibundles字符串来

String osgiBundles="";

// 配置Equinox的启动

FrameworkProperties.setProperty("osgi.noShutdown","true");

FrameworkProperties.setProperty("eclipse.ignoreApp","true");

FrameworkProperties.setProperty("osgi.bundles.defaultStartLevel","4");

FrameworkProperties.setProperty("osgi.bundles",

osgiBundlesBuilder.toString());

// 根据需要设置bundle所在的路径

String bundlePath="";

// 指定要加载的plugins所在的目录

FrameworkProperties.setProperty("osgi.syspath", bundlePath);

// 调用EclipseStarter,完成容器的启动,指定configuration目录

EclipseStarter.run(newString[]{"-configuration","configuration",

"-console"},null);

// 通过EclipeStarter获得BundleContext

context=EclipseStarter.getSystemBundleContext();

}

/**

* 停止equinox容器

*/

publicstaticvoidstop(){

try{

EclipseStarter.shutdown();

context=null;

}

catch(Exception e) {

System.err.println("停止equinox容器时出现错误:"+e);

e.printStackTrace();

}

}

/**

* 从equinox容器中获取OSGi服务instance   还可以基于此进一步处理多服务接口实现的状况

*

* @param serviceName 服务名称(完整接口类名)

*

* @return Object 当找不到对应的服务时返回null

*/

publicstaticObject getOSGiService(String serviceName){

ServiceReference serviceRef=context.getServiceReference

(serviceName);

if(serviceRef==null)

returnnull;

returncontext.getService(serviceRef);

}

/**

* 获取OSGi容器中插件的类

*/

publicstaticClass> getBundleClass(String bundleName,

String className)throwsException{

Bundle[] bundles=context.getBundles();

for(inti =0; i

if(bundleName.equalsIgnoreCase(bundles[i].getSymbolicName())){

returnbundles[i].loadClass(className);

}

}

在实现了OSGi容器外与OSGi交互之后,通常会同时产生一个需求,就是在OSGi容器内的插件要加载OSGi容器外的类,例如OSGi容器内提供了一个mvc框架,而Action类则在OSGi容器外由其他的容器负责加载,那么这个时候就会产生这个需求了,为了做到这点,有一个比较简单的解决方法,就是编写一个Bundle,在该Bundle中放置一个允许设置外部ClassLoader的OSGi服务,例如:publicclassClassLoaderService{

publicvoidsetClassLoader(ClassLoader classloader);

}

基于上面的方法,在外部启动Equinox的类中去反射执行ClassLoaderService这个OSGi服务的setClassLoader方法,将外部的classloader设置进来,然后在OSGi容器的插件中要加载OSGi容器外的类的时候就调用下面这个ClassLoaderService去完成类的加载。

基于以上说的这些方法,基本上可以较好地实现OSGi容器与其他容器的结合,例如在tomcat中启动OSGi等,或者在我们自身的应用中来控制OSGi的容器。到这里,我们基本上完成了对于Equinox的介绍,下面来看另外一个应用也较广泛的OSGi的容器--Felix。

java osgi equinox_OSGI Equinox 环境搭建相关推荐

  1. Java语言概述以及环境搭建

    Java语言概述以及环境搭建 计算机系结构 硬件 软件 人机交互方式 常见的DOS命令 Java语言的概述 Java语言的背景 Java语言的特点 Java的主要版本 JDK下载 <1>J ...

  2. 测试小白基于java的selenium自动化测试环境搭建

    本人是一个软件测试小白,自己在研究自动化测试时,真的入了不少坑,直到现在,才勉强将环境搭建了起来. 在此,我随随便便总结了一下基于java的selenium自动化测试环境搭建,送给正在学习软件测试的你 ...

  3. Java JDK 8u221开发环境搭建

    Java JDK 8u221开发环境搭建 Java JDK 8u221下载 1.java SE8u221的下载 搭建环境 更改路径 现在我们来搭建Java开发环境 Java JDK 8u221下载 1 ...

  4. Docker java项目发布/开发环境搭建

    Docker java项目发布/开发环境搭建 线上部署项目/开发环境搭建命令大全 关闭windows10 指定应用 安装Docker 安装 docker-compose 关于docker容器导致硬盘耗 ...

  5. Java+Selenium3.3.1环境搭建

    一.背景和目的 selenium从2.0开始,加入了webdriver,实际上,我们说的selenium自动化测试,大部分情况都是在使用webdriver的API.现在去Selenium官网,发现最新 ...

  6. Java及Android开发环境搭建

    前言 自从接触java以来,配置环境变量折腾了好几次,也几次被搞得晕头转向,后来常常是上网查阅相关资料才解决.但是过一段时间后一些细节就会记不清了,当要在其他机子上配置时又得上网查或者查阅相关书籍,如 ...

  7. eclipse java swt_Eclipse swt开发环境搭建

    原料: eclipse swt.下载链接为: 配置说明: 步骤: 1.下载swt,格式为.zip 2.将swt**.zip导入到eclipse中,file-->import-->Impor ...

  8. Java专题 Java使用第一步 环境搭建JDK及JRE(含多版本JDK切换思路)

    Java使用第一步 去百度上搜一下你就会发现Java是一个跨平台的语言,也就是说用java编写的程序可以在不同的操作系统上运行,简单来说就是一次编译,到处运行,相比一些其他限制操作系统类型的语言来说那 ...

  9. 第一章Java概述及开发环境搭建

    本章目标: 了解SUN公司 了解Java语言的发展史 了解Java主要技术的三个分支 了解Java可移植性的实现原理 SUN公司:java的缔造者 Java发展简史: Oak:Java的前身,1991 ...

最新文章

  1. python的自带数据集_解决Keras自带数据集与预训练model下载太慢问题
  2. VS2019如何导出exe文件_Win10+VS2019源码编译HDF5(1.10.6)
  3. Web Cookie详解
  4. 计算机病毒揭秘与对抗--注册表操作2
  5. phpcms后台系统怎么去掉html目录_电子笔记本 | 好记性胜过烂笔头?基于python3的知识管理系统...
  6. android webview设置加载进度条
  7. jQuery无任何标示获取td在表格中的行数和列数
  8. HTML 中CSS 图像详解
  9. Cookies 、 Session 和 token 的区别
  10. annotation:@Override出现The method of type must override asuperclass解决方案
  11. nexus-3.37.3 报INSTALL4J_JAVA_HOME to point to a suitable JVM
  12. K700C通过电脑共享宽带上网!
  13. WebVirtMgr + KVM 环境中的 Linux 虚拟机部署
  14. 吴忠军 - 养生之道 老人健康长寿的大秘诀
  15. 怎么才能做好一套软件系统
  16. android 免root冻结,自冻FreezeYou(超强免ROOT冻结神器)
  17. leetcode 507 完美数
  18. PINN解偏微分方程实例3(Allen-Cahn方程)
  19. 设计模式 —— Builder 模式
  20. 相机标定与矫正(总结)

热门文章

  1. 2005年GCT真题
  2. js实现title滚动效果
  3. 西安电子科技计算机科学与技术,刘志镜 - 西安电子科技大学 - 计算机科学与技术学院...
  4. secureCRT log file的设置
  5. Python中的复数类型介绍
  6. 大专简单的2年程序人生
  7. php打开rs232,PHP在RS232串口通讯协议的应用演示[测试环境WinXP/PHP5.1.4]
  8. JS控制文本框只读状态(普通、EasyUI的textbox)
  9. 【深入研究js事件】
  10. 音视频编解码:FLV封装格式笔记