`
kanwoerzi
  • 浏览: 1644589 次
文章分类
社区版块
存档分类
最新评论

基于 EventAdmin 服务的 Bundle间通讯

 
阅读更多

OSGi 的 Event Admin 服务规范提供了开发者基于发布 / 订阅模型,通过事件机制实现 Bundle 间协作的标准通讯方式。

事件发布者使用 Event Admin 服务发送基于主题 (Topic) 的事件,任何对某一主题感兴趣的事件订阅者都会收到该事件,并且做出相应的反应。

如何发布事件

我们遵循以下的步骤,通过 Event Admin 发布事件:

获得实现了 org.osgi.service.event.EventAdmin 接口的 Event Admin 服务引用。
拟定事件的主题。
指定事件所携带的 key/value 属性集。
使用 Event Admin 提供的 postEvent 或者 sendEvent 方法发布事件,postEvent 使用同步的方式发布事件,即:等到所有的事件订阅者响应了该事件,方法调用返回,而 sendEvent 使用异步的方式发布事件。
发布事件:

/**
* 自定义Bundle(服务)实现类,并且将事件发送到EventAdmin中
* @author zxpWork
*
*/
public class HelloWordImpl implements HelloWord {

public String getHello() {
BundleContext context=Activator.contet;
ServiceReference ref =context.getServiceReference(EventAdmin.class.getName()); //获取EventAdmin服务
if(null!=ref){
EventAdmin eventAdmin = (EventAdmin)context.getService(context.getServiceReference(EventAdmin.class.getName()));
if(eventAdmin!=null) {
System.out.println("event started");
// eventAdmin.postEvent(new MyEvent()); //异步 :发送事件,不管监听方是否接收成功直接返回。
eventAdmin.sendEvent(new MyEvent());//同步 :发送事件,需要所有监听方正常接收成功后才有返回。
System.out.println("event returned");
}
}
return "Hello OSGi Server!";
}


/**
* 自定义事件,继承Event
* @author zxpWork
*
*/
public class MyEvent extends Event {

public static final String MY_TOPIC="org/zxp/event1/event/MyEvent";

public MyEvent(String arg0, Dictionary arg1) {
super(arg0, arg1);
}

public MyEvent(){
super(MY_TOPIC, null);
}

public String toString(){
return "MyEvent";
}

}
}


public class Activator implements BundleActivator {

public static BundleContext contet;

private ServiceRegistration sr;

/*
* (non-Javadoc)
* @see org.osgi.framework.BundleActivator#start(org.osgi.framework.BundleContext)
*/
public void start(BundleContext context) throws Exception {
System.out.println("EventService start!");
contet=context;
sr=context.registerService(HelloWord.class.getName(), new HelloWordImpl(), null);
/*注册ServiceTracker*/
// ServiceTracker st=new ServiceTracker(context,EventAdmin.class.getName(),null);
// st.open();
}

/*
* (non-Javadoc)
* @see org.osgi.framework.BundleActivator#stop(org.osgi.framework.BundleContext)
*/
public void stop(BundleContext context) throws Exception {
System.out.println("EventService stop!");
sr.unregister();
}

}

如何订阅事件

遵循以下的步骤,来订阅被 Event Admin 发布的事件:

确定你想要订阅那些主题事件,支持通配符。
如果需要,根据事件的属性确定事件的过滤原则。
发布实现了 org.osgi.service.event.EventHandler 接口的你的事件订阅服务,以便当你感兴趣的事件发生后,Event Admin 能通知到你。
事件订阅:

/**
* 自定义事件处理(监听)类
* @author zxpWork
*
*/
public class MyEventHandler implements EventHandler {

public void handleEvent(Event event) {
System.out.println("MyEventHandle started:"+event);
try {
Thread.currentThread().sleep(5*1000);
} catch (InterruptedException e) {

}
System.out.println("MyEventHandle ok--"+event);
}

}


public class Activator implements BundleActivator {

/*
* (non-Javadoc)
*
* @see
* org.osgi.framework.BundleActivator#start(org.osgi.framework.BundleContext
* )
*/
public void start(BundleContext context) throws Exception {
System.out.println("EventHandler start!");
/*
* 添加事件处理器 (监听).
*/
String[] topics = new String[] { MyEvent.MY_TOPIC };
Hashtable<String, String[]> ht = new Hashtable<String, String[]>();
ht.put(EventConstants.EVENT_TOPIC, topics);
EventHandler myHandler = new MyEventHandler();//新建事件处理器(监听)
context.registerService(EventHandler.class.getName(), myHandler, ht);//注册事件处理器(监听)
System.out.println("event handler registered");

/*
* 获取自定义Bundle服务
*/
HelloWord hello1 = (HelloWord) context.getService(context
.getServiceReference(HelloWord.class.getName()));
System.out.println(hello1.getHello());
}

/*
* (non-Javadoc)
*
* @see
* org.osgi.framework.BundleActivator#stop(org.osgi.framework.BundleContext)
*/
public void stop(BundleContext context) throws Exception {
System.out.println("EventHandler stop!");
}

运行结果:

osgi> EventHandler start!
event handler registered
EventService start!
event started
MyEventHandle started:MyEvent
MyEventHandle ok--MyEvent
event returned
Hello OSGi Server!

从这个例子中可以看到,发布事件的 Bundle 与订阅事件的 Bundle 之间没有任何的相互依赖关系,它们之间以 Event Admin 服务为中间人 (Broker),以事件 (Event) 为消息载体,进行 Bundle 间的松散协作。

基于 Event Admin 服务的 Bundle 通讯,我们总结如下几点:

OSGi Event Admin 的事件模型,与传统的 JMS 相比,OSGi Event Admin 事件不是持久化的,也就是说,通过 Event Admin 发布的事件只在本次应用会话期间存在,当 OSGi 应用重新启动后,所有的未处理的事件将全部丢失。
由于 OSGi 服务的动态特性,使用该模型时,必须考虑到以下的情况:由于订阅事件的 Bundle 可以在任意运行时刻注册实现了 org.osgi.service.event.EventHandler 接口的事件订阅服务,所以开发者不能期望订阅者总能够收到在本次会话期间事件发布者发布的所有事件,一个典型的情况是,如果在发布者发布事件的时候,订阅 者服务尚未注册,其结果是,该订阅者错过这次事件。
在 Equinox 环境下,如果想使用 Event Admin 服务,需要额外从http://www.eclipse.org/equinox/bundles/ 下载,具体的 Event Admin 服务 Equinox 实现 org.eclipse.equinox.event Bundle, 在默认的情况下,org.eclipse.equinox.event Bundle 并没有包含在 Equinox 的 SDK 里。

小结:
OSGi 服务层所提供的服务模型,其动态发现、动态绑定、契约与实现相分离等特性,可以帮助开发者实现组件之间依赖保持在接口之间,而与具体实现代码相对松散的应 用逻辑。在此基础上 OSGi 提供了基于事件的 Event Admin 服务。Event Admin 服务本质上是基于消息的发布 / 订阅机制,对于 OSGi 的开发者来说,Event Admin 服务可以帮助他们实现组件间更为松散的,不依赖任何接口定义的协作逻辑。

分享到:
评论

相关推荐

    第五次作业函数第一题代码

    第五次作业函数第一题--

    基于深度学习的作物病害诊断内含数据集和运行环境说明.zip

    本项目旨在利用深度学习方法实现作物病害的自动诊断。作物病害是农业生产中的重要问题,及时诊断和处理对于减少产量损失至关重要。 我们采用深度学习算法,通过分析作物的图像,实现对病害的自动识别和分类。项目使用的数据集包括公开的作物病害图像数据集,如ISIC等,并进行了预处理,包括图像增强、分割和特征提取等。 在运行环境方面,我们使用Python编程语言,基于TensorFlow、PyTorch等深度学习框架进行开发。为了提高计算效率,我们还使用了GPU加速计算。此外,我们还采用了Docker容器技术,确保实验结果的可重复性。 项目完成后,将实现对作物病害的快速、准确诊断,为农业生产提供有力支持,有助于减少产量损失。同时,项目成果也可应用于其他图像识别和分类任务。

    机械设计CD驱动印刷设备step非常好的设计图纸100%好用.zip

    机械设计CD驱动印刷设备step非常好的设计图纸100%好用.zip

    tensorflow-2.7.2-cp37-cp37m-manylinux2010-x86-64.whl

    python烟花代码

    python烟花代码示例

    附件中是一个简单的烟花效果的代码示例: 在Python中,可以使用多种方式来模拟烟花效果,其中一种常用的方法是使用turtle模块,它提供了一个画布和一个小海龟,可以用来绘制各种图形。 这段代码首先导入了turtle模块和random模块,然后在屏幕上绘制了10次烟花爆炸的效果。每次爆炸都是由5个小圆组成,颜色随机选择,圆的大小也是随机的。 请注意,这段代码需要在支持turtle模块的Python环境中运行,并且需要有图形界面的支持。如果你在没有图形界面的环境中(比如某些服务器或者命令行界面),这段代码可能无法正常运行。

    商业化产品经理,到底如何实现产品商业化?.docx

    商业化产品经理,到底如何实现产品商业化?.docx

    Panduit 工业以太网部件内部销售指南

    Panduit 工业以太网部件内部销售指南

    Java版三维装箱代码示例

    在Java中,实现一个三维装箱(也称为三维背包问题)的算法通常涉及到组合优化和动态规划。这个问题是一个典型的优化问题,其中目标是在三个维度的限制下最大化价值的总和。下面是一个简单的Java代码示例,它使用动态规划来解决三维装箱问题。 请注意,这个代码只是一个简单的示例,它假设所有物品的第三个维度的大小都是1,并且没有给出如何回溯选择物品的完整逻辑。在实际应用中,三维装箱问题可能更加复杂,需要考虑所有三个维度的限制,并且可能需要更复杂的算法来解决。 此外,这个问题的解决方案可能需要根据具体问题的要求进行调整,例如物品是否可以分割、是否允许超过一个的物品等。如果你有特定的问题描述或者需要进一步的帮助,请提供更多的细节。

    常用品牌EPLAN部件库

    常用品牌EPLAN部件库

    单片机开发的教程.doc

    单片机开发的教程可以分为以下几个步骤: 1. 了解单片机基础知识:在学习单片机开发之前,需要了解单片机的相关知识,包括单片机的基本结构、指令系统、编程语言等。 2. 选择开发板:选择一款适合自己学习开发板的型号和厂商,通常需要关注开发板的性价比、开发环境是否友好等因素。 3. 学习开发环境:根据所选的开发板,学习相关的开发环境和使用方法,例如Keil、IAR等集成开发环境。 4. 掌握编程语言:单片机常用的编程语言包括C语言和汇编语言,根据实际情况选择其中一种进行学习。 5. 基础操作:熟悉单片机的引脚定义和IO口配置,了解单片机的启动代码,可以通过修改启动代码进行基本功能调试。 6. 综合实践:根据具体项目需求,进行单片机开发的综合实践。在实践中需要掌握如何编写程序、如何进行硬件调试、如何使用相关工具软件等技能。 下面是一个单片机开发的简单教程介绍: 首先,确定所使用的单片机型号和开发板类型。在这个阶段,需要查阅相关资料,了解开发板的规格书、芯片规格等基本资料。 其次,安装并配置开发环境。根据所选的开发板,安装相应的集成开发环境(IDE),并配置好开发环境。 接着,学习并掌

    Q1.ipynb

    Q1.ipynb

    (自适应手机端)IT网络建站公司pbootcms模板 互联网营销企业网站源码下载.zip

    (自适应手机端)IT网络建站公司pbootcms模板 互联网营销企业网站源码下载.zip

    Bematech 激光扫描器用户手册

    Bematech 激光扫描器用户手册

    激励视频接入文档.pdf

    激励视频接入文档.pdf

    java jdk1.8 202版本下载window linux打包

    java jdk1.8 202版本下载window linux打包

    Lite Beam M5快速指南 Lite Beam M5天线设置指南

    Lite Beam M5快速指南

    互联网金融导论.docx

    互联网金融导论.docx

    字节跳动青训营-抖音项目

    字节跳动青训营——抖音项目

    node-v12.22.10-linux-s390x.tar.xz

    Node.js,简称Node,是一个开源且跨平台的JavaScript运行时环境,它允许在浏览器外运行JavaScript代码。Node.js于2009年由Ryan Dahl创立,旨在创建高性能的Web服务器和网络应用程序。它基于Google Chrome的V8 JavaScript引擎,可以在Windows、Linux、Unix、Mac OS X等操作系统上运行。 Node.js的特点之一是事件驱动和非阻塞I/O模型,这使得它非常适合处理大量并发连接,从而在构建实时应用程序如在线游戏、聊天应用以及实时通讯服务时表现卓越。此外,Node.js使用了模块化的架构,通过npm(Node package manager,Node包管理器),社区成员可以共享和复用代码,极大地促进了Node.js生态系统的发展和扩张。 Node.js不仅用于服务器端开发。随着技术的发展,它也被用于构建工具链、开发桌面应用程序、物联网设备等。Node.js能够处理文件系统、操作数据库、处理网络请求等,因此,开发者可以用JavaScript编写全栈应用程序,这一点大大提高了开发效率和便捷性。 在实践中,许多大型企业和组织已经采用Node.js作为其Web应用程序的开发平台,如Netflix、PayPal和Walmart等。它们利用Node.js提高了应用性能,简化了开发流程,并且能更快地响应市场需求。

    node-v12.12.0-linux-x64.tar.xz

    Node.js,简称Node,是一个开源且跨平台的JavaScript运行时环境,它允许在浏览器外运行JavaScript代码。Node.js于2009年由Ryan Dahl创立,旨在创建高性能的Web服务器和网络应用程序。它基于Google Chrome的V8 JavaScript引擎,可以在Windows、Linux、Unix、Mac OS X等操作系统上运行。 Node.js的特点之一是事件驱动和非阻塞I/O模型,这使得它非常适合处理大量并发连接,从而在构建实时应用程序如在线游戏、聊天应用以及实时通讯服务时表现卓越。此外,Node.js使用了模块化的架构,通过npm(Node package manager,Node包管理器),社区成员可以共享和复用代码,极大地促进了Node.js生态系统的发展和扩张。 Node.js不仅用于服务器端开发。随着技术的发展,它也被用于构建工具链、开发桌面应用程序、物联网设备等。Node.js能够处理文件系统、操作数据库、处理网络请求等,因此,开发者可以用JavaScript编写全栈应用程序,这一点大大提高了开发效率和便捷性。 在实践中,许多大型企业和组织已经采用Node.js作为其Web应用程序的开发平台,如Netflix、PayPal和Walmart等。它们利用Node.js提高了应用性能,简化了开发流程,并且能更快地响应市场需求。

Global site tag (gtag.js) - Google Analytics