首页 > 解决方案 > 在 Eclipse RCP 中注入单例 OSGi 声明式服务

问题描述

我正在尝试定义一个单例 OSGi 服务,该服务将由我的 Eclipse RCP 应用程序中的其他插件使用(=共享),但每个插件都有自己的版本(来自本地类加载器),并且只有排名最高的一个版本被注入。

该服务是这样定义的(在com.test.taskmodel捆绑中)(暂时不使用单独的接口和实现):

@Component(scope=ServiceScope.SINGLETON, service=TaskService.class)
public final class TaskService {

    public TaskService() {}

    @Activate
    void activate(BundleContext bundleContext) {
        //activate
    }
    
    public Result doStuff() {
        return null;
    }
}

当我使用 bnd 构建它时,生成的 jar 清单包含Service-Component: OSGI-INF/com.test.TaskService.xml并且 xml 是:

<?xml version="1.0" encoding="UTF-8"?>
<scr:component xmlns:scr="http://www.osgi.org/xmlns/scr/v1.3.0" name="com.test.TaskService" activate="activate" deactivate="deactivate">
  <service scope="singleton">
    <provide interface="com.test.TaskService"/>
  </service>
  <implementation class="com.test.TaskService"/>
</scr:component>

有三个 Eclipse RCP 插件使用该服务taskbrowsertaskfiltertaskdetail​​ . 它们现在几乎都是相同的,每个只有一个像这样的部分:

public class TaskBrowserPart {
    
    @Inject @Service @Optional private TaskService taskService;

    @PostConstruct
    public void createControls(Composite parent) {
        if (taskService == null) {
            System.out.println("TaskBrowser TaskService null");
        } else {
            System.out.println("TaskBrowser TaskService non-null");
        }
    }
}

当我运行应用程序时,我得到以下输出:

TaskBrowser TaskService non-null
TaskFilter TaskService null
TaskDetail TaskService null

在 osgi 控制台中调用service命令会显示四个已注册的服务,只有第一个被所有树插件使用:

{com.test.taskmodel.TaskService}={service.id=63, service.bundleid=155, service.scope=bundle, component.name=com.test.taskmodel.TaskService, component.id=34}
  "Registered by bundle:" com.test.taskbrowser [155]
  "Bundles using service"
    com.test.taskfilter [165]
    com.test.taskdetail [158]
    com.test.taskbrowser [155]
{com.test.taskmodel.TaskService}={service.id=65, service.bundleid=158, service.scope=bundle, component.name=com.test.taskmodel.TaskService, component.id=36}
  "Registered by bundle:" com.test.taskdetail [158]
  "No bundles using service."
{com.test.taskmodel.TaskService}={service.id=66, service.bundleid=159, service.scope=bundle, component.name=com.test.taskmodel.TaskService, component.id=37}
  "Registered by bundle:" com.test.taskmodel [159]
  "No bundles using service."
{com.test.taskmodel.TaskService}={service.id=87, service.bundleid=165, service.scope=bundle, component.name=com.test.taskmodel.TaskService, component.id=39}
  "Registered by bundle:" com.test.taskfilter [165]
  "No bundles using service."

service.scope=bundle当我定义了服务时,为什么会有多个服务实例scope="singleton"?如何以仅存在一个服务实例并且所有插件都使用它的方式定义它?为什么 Eclipse 尝试只注入第一个服务,然后在taskfiltertaskdetail插件中失败,因为它们使用不同的类加载器,因此服务类型不匹配?为什么选择服务时不考虑类加载器,而只在注入时考虑?在那种情况下,我至少在每个插件中都有一个单独的实例,这不是我想要的,但至少是我可以使用的。任何帮助表示赞赏,谢谢。

标签: javaosgieclipse-rcpbndbndtools

解决方案


您要完成的实际上是声明式服务的默认行为。您不需要指定 ServiceScope.SINGLETON 属性,也不需要在此处使用额外的 @Service 注释(尽管这些都不会导致问题)。

这是否发生在使用工具创建 DS XML 文件的 Eclipse IDE 本身中?我注意到您在问题中使用 Bnd 引用。您是否看到任何旧的 XML 文件或服务组件条目?或者 OSGi 缓存中是否有可能存在陈旧数据?

假设 Service-Component 条目仅在 TaskModel 包中,您真的不应该看到创建 TaskService 的 TaskBrowser、TaskDetail 和 TaskFilter 包。


推荐阅读