我想用OSGI完成一些事情,这在像Guice或Spring这样的依赖注入框架中是非常简单的。
我想将服务实现类放在与服务接口相同的包中。这使我能够使用与服务客户无关的包可见方法。
Guice实现此目的的方法很简单,使服务实现类包可见。服务界面:
package com.example.services;
public interface SomeService {
void scanClasses(ServiceHelper helper);
}
实施:
package com.example.services;
// Package visible service implementation
class SomeServiceImpl implements SomeService {
@Override
public void scanClasses(ServiceHelper helper) {
ClassLoader bundlesClassLoader = helper.getClassLoader();
// do something with bundle's classes
}
}
这是ServiceHelper类,它也有包可见方法:
package com.example.services;
import org.osgi.framework.BundleContext;
import org.osgi.framework.wiring.BundleWiring;
public class ServiceHelper {
private ClassLoader bundleClassLoader;
public ServiceHelper(BundleContext bc) {
this.bundleClassLoader = bc.getBundle().adapt(BundleWiring.class).getClassLoader();
}
// Package visible method to be called from SomeServiceImpl class
ClassLoader getClassLoader() {
return bundleClassLoader;
}
}
但OSGI无法使用包可见的decleration或构造函数实例化ServiceImpl类:
java.lang.IllegalAccessException:类org.eclipse.equinox.internal.ds.model.ServiceComponent无法使用修饰符访问com.example.SomeServiceImpl类的成员""
要完成该示例,这是服务客户端代码,不应受设计影响:
package com.example.serviceclient;
import org.osgi.framework.BundleContext;
import com.example.services.ServiceHelper;
import com.example.services.SomeService;
public class ServiceClientExample {
private SomeService someService;
public void activate(BundleContext bc) {
someService.scanClasses(new ServiceHelper(bc));
}
public void setSomeService(SomeService service) {
this.someService = service;
}
}
将服务实现放到另一个包中,并尝试构建良好的面向对象的封装需要太多的工作,比如将包可见方法映射到调用者的访问器类,我相信应该有另一种方式。
答案 0 :(得分:2)
您正在使用OSGi声明式服务,它确实要求实现类是公共的并且具有公共的零参数构造函数。这是设计的。你可以通过使用BundleActivator
并在代码中实例化组件来解决这个问题,但是没有什么意义......
如果您的实现类与接口位于同一个包中,则接口的每个客户端都直接耦合到实现类。因此,您可以将实现类公之于众,并允许客户端直接实例化它。
OSGi允许合同,提供商和消费者的真正分离。将合同和提供商合并到同一个包中就会失败。 Guice和Spring将这些概念结合在一起“直截了当”的事实既不存在也不存在。