我有以下课程:
@Component
@ConifgurationProperties("redis")
public class RedisProperties {
private List<String> hosts;
// getters, setters
}
@Component
public class StaticRedisHostsProvider implements RedisHostsProvider {
private final RedisProperties redisProperties;
public StaticRedisHostsProvider(RedisProperties redisProperties) {
this.redisProperties = redisProperties;
}
@Override
public List<String> getAll() {
return redisProperties.getHosts();
}
}
@Component
public DiscoveryBasedRedisHostsProvider { ... }
如果指定了StaticRedisHostsProvider
属性,我希望使用redis.hosts
,否则DiscoveryBasedRedisHostsProvider
。
我可以使用StaticRedisHostsProvider
注释@ConditionalOnProperty(prefix = "redis", name = "hosts")
,但没有与@ConditionalOnMissingProperty
一起使用的类似DiscoveryBasedRedisHostsProvider
注释。
我尝试使用@ConditionalOnExpression("@redisProperties.hosts.empty")
,但由于某种原因它无法运作:
Description:
A component required a bean named 'redisProperties' that could not be found.
Action:
Consider defining a bean named 'redisProperties' in your configuration.
有没有办法解决这个问题(可能有@Order
或类似的注释)?
答案 0 :(得分:1)
您必须添加matchIfMissing
以检查是否没有属性
@ConditionalOnProperty(prefix = "redis", name = "hosts", matchIfMissing = true)
默认设置为false
修改强>
如果您想检查是否未设置该值,则可以尝试使用havingValue=""
。
@ConditionalOnProperty(prefix = "redis", name = "hosts", havingValue="")
答案 1 :(得分:0)
这是我在Spring自动配置中使用自定义条件的问题。
@Conditional
批注在应用程序启动期间非常早地执行。属性源已经加载,但是尚未创建ConfgurationProperties Bean。但是,我们可以通过将属性自己绑定到Java POJO来解决该问题。
首先,我介绍一个功能接口,该接口使我们能够定义任何自定义逻辑检查属性是否存在。在您的情况下,此方法将检查属性List是否为空或空。
public interface OptionalProperties {
boolean isPresent();
}
现在让我们创建一个注释,该注释将使用Spring @Conditional
进行元注释,并允许我们定义自定义参数。 prefix
代表属性名称空间,targetClass
代表应将属性映射到的配置属性模型类。
@Target({ElementType.TYPE, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Conditional(OnConfigurationPropertiesCondition.class)
public @interface ConditionalOnConfigurationProperties {
String prefix();
Class<? extends OptionalProperties> targetClass();
}
现在是主要部分。自定义条件实现。
public class OnConfigurationPropertiesCondition extends SpringBootCondition {
@Override
public ConditionOutcome getMatchOutcome(ConditionContext context, AnnotatedTypeMetadata metadata) {
MergedAnnotation<ConditionalOnConfigurationProperties> mergedAnnotation = metadata.getAnnotations().get(ConditionalOnConfigurationProperties.class);
String prefix = mergedAnnotation.getString("prefix");
Class<?> targetClass = mergedAnnotation.getClass("targetClass");
// type precondition
if (!OptionalProperties.class.isAssignableFrom(targetClass)) {
return ConditionOutcome.noMatch("Target type does not implement the OptionalProperties interface.");
}
// the crux of this solution, binding properties to Java POJO
Object bean = Binder.get(context.getEnvironment()).bind(prefix, targetClass).orElse(null);
// if properties are not present at all return no match
if (bean == null) {
return ConditionOutcome.noMatch("Binding properties to target type resulted in null value.");
}
OptionalProperties props = (OptionalProperties) bean;
// execute method from OptionalProperties interface
// to check if condition should be matched or not
// can include any custom logic using property values in a type safe manner
if (props.isPresent()) {
return ConditionOutcome.match();
} else {
return ConditionOutcome.noMatch("Properties are not present.");
}
}
}
现在,您应该创建实现OptionalProperties
接口的自己的配置属性类。
@ConfigurationProperties("redis")
@ConstructorBinding
public class RedisProperties implements OptionalProperties {
private final List<String> hosts;
@Override
public boolean isPresent() {
return hosts != null && !hosts.isEmpty();
}
}
然后在Spring @Configuration
类中。
@Configuration
class YourConfiguration {
@ConditionalOnConfigurationProperty(prefix = "redis", targetClass = RedisProperties.class)
StaticRedisHostsProvider staticRedisHostsProvider() {
...
}
@ConditionalOnMissingBean(StaticRedisHostsProvider.class)
DiscoveryBasedRedisHostsProvider discoveryRedisHostsProvider() {
...
}
}
此解决方案有两个缺点:
@ConfigurationProperties
注释上和在@ConditionalOnConfigurationProperties
注释上。通过在配置属性POJO中定义public static final String PREFIX = "namespace"
,可以部分缓解这种情况。