`
yanxia611
  • 浏览: 42045 次
  • 性别: Icon_minigender_2
  • 来自: 北京
社区版块
存档分类
最新评论

从外部启动Equinox

    博客分类:
  • java
阅读更多
从外部启动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容器交互的类可以这么写:



Java代码
/**   
  * 启动并运行equinox容器   
  */   
public static void start() throws Exception{    
     // 根据要加载的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(new String[]{"-configuration","configuration",    
        "-console"}, null);    
     // 通过EclipeStarter获得BundleContext    
     context=EclipseStarter.getSystemBundleContext();    
}    
     
/**   
* 停止equinox容器   
*/   
public static void stop(){    
    try {    
        EclipseStarter.shutdown();    
        context=null;    
    }    
    catch (Exception e) {    
             System.err.println("停止equinox容器时出现错误:"+e);    
             e.printStackTrace();    
     }    
}    
   
/**   
  * 从equinox容器中获取OSGi服务instance   还可以基于此进一步处理多服务接口实现的状况   
  *    
  * @param serviceName 服务名称(完整接口类名)   
  *    
  * @return Object 当找不到对应的服务时返回null   
  */   
public static Object getOSGiService(String serviceName){    
     ServiceReference serviceRef=context.getServiceReference    
       (serviceName);    
     if(serviceRef==null)    
         return null;    
     return context.getService(serviceRef);    
}    
     
/**   
  * 获取OSGi容器中插件的类   
  */   
public static Class<?> getBundleClass(String bundleName,    
   String className) throws Exception{    
     Bundle[] bundles=context.getBundles();    
     for (int i = 0; i < bundles.length; i++) {    
        if(bundleName.equalsIgnoreCase(bundles[i].getSymbolicName())){    
             return bundles[i].loadClass(className);    
     }    


/** 
  * 启动并运行equinox容器 
  */
public static void start() throws Exception{ 
     // 根据要加载的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(new String[]{"-configuration","configuration", 
        "-console"}, null); 
     // 通过EclipeStarter获得BundleContext 
     context=EclipseStarter.getSystemBundleContext(); 

  
/** 
* 停止equinox容器 
*/
public static void stop(){ 
    try { 
        EclipseStarter.shutdown(); 
        context=null; 
    } 
    catch (Exception e) { 
             System.err.println("停止equinox容器时出现错误:"+e); 
             e.printStackTrace(); 
     } 


/** 
  * 从equinox容器中获取OSGi服务instance   还可以基于此进一步处理多服务接口实现的状况 
  *  
  * @param serviceName 服务名称(完整接口类名) 
  *  
  * @return Object 当找不到对应的服务时返回null 
  */
public static Object getOSGiService(String serviceName){ 
     ServiceReference serviceRef=context.getServiceReference 
       (serviceName); 
     if(serviceRef==null) 
         return null; 
     return context.getService(serviceRef); 

  
/** 
  * 获取OSGi容器中插件的类 
  */
public static Class<?> getBundleClass(String bundleName, 
   String className) throws Exception{ 
     Bundle[] bundles=context.getBundles(); 
     for (int i = 0; i < bundles.length; i++) { 
        if(bundleName.equalsIgnoreCase(bundles[i].getSymbolicName())){ 
             return bundles[i].loadClass(className); 
     } 
}      在实现了OSGi容器外与OSGi交互之后,通常会同时产生一个需求,就是在OSGi容器内的插件要加载OSGi容器外的类,例如OSGi容器内提供了一个mvc框架,而Action类则在OSGi容器外由其他的容器负责加载,那么这个时候就会产生这个需求了,为了做到这点,有一个比较简单的解决方法,就是编写一个Bundle,在该Bundle中放置一个允许设置外部ClassLoader的OSGi服务,例如:



Java代码
public class ClassLoaderService{    
    public void setClassLoader(ClassLoader classloader);    
}  

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

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

相关推荐

    OSGi™ Component Programming

    - **生命周期管理**:组件有安装(Installed)、解析(Resolved)、启动(Started)、停止(Stopped)和卸载(Uninstalled)等状态,框架控制这些状态的转换。 - **类加载器**:每个包都有自己的类加载器,负责加载...

    eclipse3[1].4.2注意事项

    在Eclipse中,链接文件(`.link`)是一种指示Eclipse从外部位置加载插件的方法。例如,`xfire.link`可以被放置于`eclipse/dropins/`目录下,用于指向外部的插件或库,从而避免了重复复制文件的麻烦,提高了效率。 #...

    Eclipse RCP Plug-in开发自学教程(Eclipse3.6)

    教程内容涵盖从基础到进阶的多个方面,每个章节尽量独立,便于学习和查阅。 1. **富客户端平台** - **概述**:介绍Eclipse RCP的概念,它是基于插件的架构,提供丰富的用户界面和强大的功能。 - **ECLIPSE RCP ...

    OSGI_Example

    随着互联网的迅猛发展,个人对网络的需求已经从单纯的工作环境扩展到了家庭生活,催生了数字家庭的概念。在这个概念中,家用网关(Residential Gateway,简称RG)作为内外网络沟通的桥梁,扮演着至关重要的角色。它...

    Spring Live中文

    - 反转控制(IoC)是一种设计模式,它允许将对象的创建、配置和管理委托给外部容器。 **暴露bean定义:** - Spring 容器通过配置文件或注解暴露bean的定义,允许开发者通过名称获取bean实例。 **配置域属性和依赖...

    SPRING DM 参考手册

    - **OSGi环境**:需要一个支持OSGi标准的运行环境,如Apache Felix或Equinox。 - **Spring框架**:需要安装和配置Spring框架及其相关的库。 - **Java环境**:需要支持Java SE 5及以上版本。 ##### 1.3 新特性介绍 ...

Global site tag (gtag.js) - Google Analytics