本范例主要演示如下技术点:
1、从OSGi外部启动OSGi容器
2、在OSGi容器内访问外部的资源文件
3、在OSGi容器内访问外部的类方法(通过反射机制)
关键点:
开发一个单独的Bundle,提供一个供外部访问的OSGi服务,外部程序在启动OSGi容器时通过反射的方式将外部的ClassLoader设置到该Bundle中,供其他Bundle访问外部资源用。
一、开发单独Bundle
1、接口及实现类
public interface ClassLoaderService {
public ClassLoader getClassLoader();
}
public class ClassLoaderServiceImpl implements com.cjm.osgi.common.service.ClassLoaderService {
private ClassLoader classLoader;
public ClassLoader getClassLoader() {
return classLoader;
}
public void setClassLoader(ClassLoader classLoader) {
this.classLoader = classLoader;
//## 以下代码仅供参考,在实际的应用中不应该出现在此方法中 ##
try{
//读取OSGi容器外的资源文件
BufferedInputStream stream = new BufferedInputStream(classLoader.getResourceAsStream("readme.txt"));
byte[] b = new byte[1024];
int i;
while((i=stream.read(b))!=-1){
System.out.println(new String(b, 0, i));
}
//通过反射调用OSGi容器外的类方法
Class clazz = classLoader.loadClass("OuterService");
Object obj = clazz.newInstance();
Method method = clazz.getMethod("sayHello", String.class);
Object result = method.invoke(obj, "cjm");
System.out.println("sayHello result: " + result);
}catch(Exception ex){
ex.printStackTrace();
}
}
}
2、Activator类
public class ClassLoaderBundleActivator implements BundleActivator {
private ServiceRegistration serviceRegistration;
public void start(BundleContext context) throws Exception {
//注册服务
serviceRegistration = context.registerService(ClassLoaderService.class.getName(), new ClassLoaderServiceImpl(), null);
System.out.println(">>>>> start " + context.getBundle().getSymbolicName());
}
public void stop(BundleContext context) throws Exception {
//注销服务
serviceRegistration.unregister();
}
}
二、外部程序启动OSGi容器
public class Test {
public void run(){
try{
//要加载的bundles
String osgiBundles = "org.eclipse.osgi_3.3.0.v20070530.jar@1:start,com.cjm.osgi.common.ClassLoaderBundle_1.0.0.jar@2:start";
//## 启动Equinox的配置 ##
//重要:缺省情况下,Equinox启动后马上shutdown,通过该参数配置Equinox启动后不关闭
FrameworkProperties.setProperty("osgi.noShutdown", "true");
//重要:让Equinox不检查eclipse.product和eclipse.application配置
FrameworkProperties.setProperty("eclipse.ignoreApp", "true");
FrameworkProperties.setProperty("osgi.bundles.defaultStartLevel", "4");
FrameworkProperties.setProperty("osgi.bundles", osgiBundles);
//bundles所在路径
String bundlePath = "./lib";
FrameworkProperties.setProperty("osgi.syspath", bundlePath);
//启动OSGi容器
EclipseStarter.run(new String[]{"-configuration", "configuration", "-console"}, null);
//取得BundleContext
BundleContext bundleContext = EclipseStarter.getSystemBundleContext();
if(bundleContext!=null){
for(int i=0;i<bundleContext.getBundles().length;i++){
Bundle bundle = bundleContext.getBundles()[i];
//通过反射将OSGi外部的ClassLoader对象设置到ClassLoaderBundle Bundle中
//这样,就可以在OSGi容器内部加载OSGi外部的资源了
if(bundle.getSymbolicName().equalsIgnoreCase("com.cjm.osgi.common.ClassLoaderBundle")){
Class clazz = bundle.loadClass(ClassLoaderServiceImpl.class.getName());
Object obj = clazz.newInstance();
Method method = clazz.getMethod("setClassLoader", ClassLoader.class);
method.invoke(obj, this.getClass().getClassLoader());
}
}
}else{
System.out.println("bundleContext is null");
}
while(true){
Thread.sleep(10000);
}
}catch(Exception ex){
ex.printStackTrace();
}finally{
try{
EclipseStarter.shutdown();
}catch(Exception e){
e.printStackTrace();
}
}
}
public static void main(String[] args) {
Test t = new Test();
t.run();
}
}
Equinox的启动参数:
http://help.eclipse.org/helios/index.jsp?topic=/org.eclipse.platform.doc.isv/reference/misc/runtime-options.html
分享到:
相关推荐
12.2.4 和任务管理组件的交互 89 12.3 人工任务管理接口 90 12.3.1 Eclipse integration 90 12.3.2 基于Web的任务视图 90 第十三章 特定领域的流程 90 13.1 介绍 90 13.2 示例:通告 91 13.2.1 创建工作定义 92 ...
12.2.4 和任务管理组件的交互 83 12.3 人工任务管理接口 84 12.3.1 Eclipse integration 84 12.3.2 基于Web的任务视图 84 第十三章 特定领域的流程 84 13.1 介绍 84 13.2 示例:通告 85 13.2.1 创建工作定义 86 ...
12.2.4 和任务管理组件的交互 83 12.3 人工任务管理接口 84 12.3.1 Eclipse integration 84 12.3.2 基于Web的任务视图 84 第十三章 特定领域的流程 84 13.1 介绍 84 13.2 示例:通告 85 13.2.1 创建工作定义 86 ...
讲解了虚拟机的热点探测方法、HotSpot的即时编译器、编译触发条件,以及如何从虚拟机外部观察和分析JIT编译的数据和结果;第五部分探讨了Java实现高效并发的原理,包括JVM内存模型的结构和操作;原子性、可见性和...
你定义的Java源代码的语法规则,直接,没有必要专门编写和维护,外部语法文件。同时保持蒸提供全面的支持,无缝集成的IDE(语法着色,代码导航,重构等)的语法和操作代码完全分离。最大限度地减少时间和费用开发...
你定义的Java源代码的语法规则,直接,没有必要专门编写和维护,外部语法文件。同时保持蒸提供全面的支持,无缝集成的IDE(语法着色,代码导航,重构等)的语法和操作代码完全分离。最大限度地减少时间和费用开发...
你定义的Java源代码的语法规则,直接,没有必要专门编写和维护,外部语法文件。同时保持蒸提供全面的支持,无缝集成的IDE(语法着色,代码导航,重构等)的语法和操作代码完全分离。最大限度地减少时间和费用开发...
你定义的Java源代码的语法规则,直接,没有必要专门编写和维护,外部语法文件。同时保持蒸提供全面的支持,无缝集成的IDE(语法着色,代码导航,重构等)的语法和操作代码完全分离。最大限度地减少时间和费用开发...
你定义的Java源代码的语法规则,直接,没有必要专门编写和维护,外部语法文件。同时保持蒸提供全面的支持,无缝集成的IDE(语法着色,代码导航,重构等)的语法和操作代码完全分离。最大限度地减少时间和费用开发...
你定义的Java源代码的语法规则,直接,没有必要专门编写和维护,外部语法文件。同时保持蒸提供全面的支持,无缝集成的IDE(语法着色,代码导航,重构等)的语法和操作代码完全分离。最大限度地减少时间和费用开发...
你定义的Java源代码的语法规则,直接,没有必要专门编写和维护,外部语法文件。同时保持蒸提供全面的支持,无缝集成的IDE(语法着色,代码导航,重构等)的语法和操作代码完全分离。最大限度地减少时间和费用开发...
你定义的Java源代码的语法规则,直接,没有必要专门编写和维护,外部语法文件。同时保持蒸提供全面的支持,无缝集成的IDE(语法着色,代码导航,重构等)的语法和操作代码完全分离。最大限度地减少时间和费用开发...
你定义的Java源代码的语法规则,直接,没有必要专门编写和维护,外部语法文件。同时保持蒸提供全面的支持,无缝集成的IDE(语法着色,代码导航,重构等)的语法和操作代码完全分离。最大限度地减少时间和费用开发...
你定义的Java源代码的语法规则,直接,没有必要专门编写和维护,外部语法文件。同时保持蒸提供全面的支持,无缝集成的IDE(语法着色,代码导航,重构等)的语法和操作代码完全分离。最大限度地减少时间和费用开发...
你定义的Java源代码的语法规则,直接,没有必要专门编写和维护,外部语法文件。同时保持蒸提供全面的支持,无缝集成的IDE(语法着色,代码导航,重构等)的语法和操作代码完全分离。最大限度地减少时间和费用开发...
你定义的Java源代码的语法规则,直接,没有必要专门编写和维护,外部语法文件。同时保持蒸提供全面的支持,无缝集成的IDE(语法着色,代码导航,重构等)的语法和操作代码完全分离。最大限度地减少时间和费用开发...
你定义的Java源代码的语法规则,直接,没有必要专门编写和维护,外部语法文件。同时保持蒸提供全面的支持,无缝集成的IDE(语法着色,代码导航,重构等)的语法和操作代码完全分离。最大限度地减少时间和费用开发...
在有状态SessionBean中,用累加器,以对话状态存储起来,创建EJB对象,并将当前的计数器初始化,调用每一个EJB对象的count()方法,保证Bean正常被激活和钝化,EJB对象是用完毕,从内存中清除…… Java Socket 聊天...
在有状态SessionBean中,用累加器,以对话状态存储起来,创建EJB对象,并将当前的计数器初始化,调用每一个EJB对象的count()方法,保证Bean正常被激活和钝化,EJB对象是用完毕,从内存中清除…… Java Socket 聊天...