当我在 DS 中使用引用时,我可以指定一个目标,让我可以缩小我想要的服务实例的范围。问题是所有示例都显示了必须在代码时完成的静态查询。有没有办法进行动态查询(也许从配置管理中提取一个属性)?
如果 DS 不支持这个,是否有另一个 OSGi 依赖注入框架(蓝图、iPojo 等)支持这个?
当我在 DS 中使用引用时,我可以指定一个目标,让我可以缩小我想要的服务实例的范围。问题是所有示例都显示了必须在代码时完成的静态查询。有没有办法进行动态查询(也许从配置管理中提取一个属性)?
如果 DS 不支持这个,是否有另一个 OSGi 依赖注入框架(蓝图、iPojo 等)支持这个?
您始终可以使用引用的目标属性来执行第一级过滤器。如果您的绑定方法具有签名
void <method-name>(<parameter-type>, Map);
然后,您可以在包含服务属性的地图上运行任何动态过滤器。如果过滤器不匹配,您可以忽略该组件(暂时。)
在替代方案中,由于组件的配置可以包含引用的目标过滤器,因此您可以修改组件的配置以更改目标过滤器。
我使用以下技巧。如果您在服务引用上指定“target”属性但将其值保留为空字符串,则在运行时将使用具有相同名称但后缀为“.target”的组件属性。
在下面的示例中,我通过我的 Karaf 容器中的 .cfg 文件动态选择我的 JDBC 源。“datasourcefactory.target”属性会自动注入到“datasourcefactory”引用的“target”属性中。
警告:我实际上不知道这个技巧是官方支持还是只是 Felix SCR 功能。我一直想在规范中查找它,看看是否提到它... +1 任何澄清其合法性的评论!
@Component(
name = "...",
specVersion = "1.1",
policy = ConfigurationPolicy.REQUIRE,
metatype = true
)
@Property(name = "dataSourceFactory.target",
value = "",
label = "dataSourceFactory target",
description = "An OSGi service filter to select the data source provider. "+
"For example: '(&(osgi.jdbc.driver.name=derby)(objectClass=org.osgi.service.jdbc.DataSourceFactory))' where 'derby' is the important bit and the rest is boilerplate. "+
"See DataSourceFactory.OSGI_JDBC_DRIVER_(NAME,CLASS,VERSION)."
)
@Reference(
name = "dataSourceFactory",
referenceInterface = org.osgi.service.jdbc.DataSourceFactory.class,
cardinality = ReferenceCardinality.MANDATORY_UNARY,
target = "", // must be overwritten by configuration property named "dataSourceFactory.target"
bind = "bindDataSourceFactory",
unbind = "unbindDataSourceFactory"
)