工作原因接触了osgi,写代码的时候遇到了service的问题,需要手动注册服务,停止服务,销毁服务,于是就学习了一下,在此记录,以便后期查阅。
起因:项目需要使用几个后台运行的调度服务,并且需要灵活地对这些服务进行操作,包括动态的修改一个服务,停止,重启一个服务。
零. 桥梁
在osgi框架中,服务主要负责模块间的交流和通信。可以使用BundleContext来指定需要执行操作的模块,并使用BundleContext来对服务进行注册和操作。为了安全性,在osgi中只有创建服务的模块对它创建的服务拥有操作的权限,其他的模块只能实现查询,而无法对其操作。
使用以下代码获取BundleContext
BundleContext bundleContext = FrameworkUtil.getBundle(class1.class).getBundleContext();
参数是这个bundle的类名。
一. 注册服务
使用上面得到的BundleContext就可以将这个服务注册到该bundle下了。使用如下代码。
context.registerService(Job.class.getName(),new Job(), prop) ;
其中,第一个参数是所要创建服务的接口名,第二个参数是该服务的实例,第三个参数是该服务的参数,dictionary型
在执行上述方法后就可以在osgi控制台通过输入services命令查看到所有的服务,当然也包括我们刚才注册的这个服务。
二. 移除服务
当bundle停止的时候,osgi框架会自动移除服务,不过也可以手动移除服务。通过如下方法:
serviceRegistration.unregister();
serviceRegistration是BundleContext.registerService()返回的对象。由于osgi不允许隔bundle移除服务,所以,只能在注册这个服务的bundle内部进行服务的移除。可以在注册服务的时候将注册服务返回的serviceRegistration对象缓存起来,然后在需要的时候对他们进行操作。
public static Map<String,ServiceRegistration> jobServiceRegistration =new HashMap<>();
三. 查找服务
首先,你需要得到当前的BundleContext。使用bundleContext.getServiceReferences(Service.class,filter);来获取服务对象的间接引用。这个方法的返回类型是ServiceReference,它可以在bundle之间互享,因为它和使用服务的bundle的生命周期无关。
为什么要用间接引用而不直接返回那个实际的服务对象呢?实际上是为了将服务的使用和服务的实现进行解耦,将服务注册表作为两者的中间人,达到跟踪和控制服务的目的,同时还可以在服务消失了以后通知使用者。
需要在意的一点,ServiceReference没有提供对服务的操作,如停止或删除。只是为了安全考虑,因为ServiceReference可以在其他的bundle中调用,而其他bundle是无法得知该服务在本模块的状态的,如果本bundle停止,服务也随之停止。这时如果其他bundle可以对该服务操作的话会引起错误。
bundleContext.getServiceReferences(Service.class,filter);方法返回ServiceReference,然后可以通过bundleContext.getService(jobService);方法获取到要查找的服务实例,其中jobService是上面获取到的ServiceReference。具体代码可以参考下面,由于服可能能有多个实例,所以使用集合存储返回的ServiceReference。
Collection<ServiceReference<Job>>ref =null;
try {
ref =bundleContext.getServiceReferences(Job.class,"(sid=123)");
} catch (InvalidSyntaxException e) {
e.printStackTrace();
}
if(ref !=null){
for (ServiceReference jobService:ref) {
Job job =bundleContext.getService(jobService);
jobServiceList.add(job);
}
}
bundleContext.getServiceReferences的第二个参数是通配符,用来过滤服务,过滤规则可以参考以下规则。
属性匹配:(vendor=Apache)、(count>3)
通配符:(vendor=Apache*)
判断某个属性是否存在:(vendor=)
条件非:(!(vendor=Apache))
条件与:(&(objectClass=com.edu.osgi.user.IUserService)(type=1))
条件或:(|(type=1)(type=2))