在开发Spring Boot应用时会用到根据条件来向Spring IoC容器注入Bean。好比配置文件存在了某个配置属性才注入Bean :java
图中红色的部分是说,只有ali.pay.v1.app-id
存在于Spring的环境配置中时这个@Configuration
标记的类才能注入Spring IoC。git
这里面的@ConditionalOnProperty
就是条件注解系列的一种。它还有不少种来知足各类场景的条件注解:spring
其实数量远不止截图中这几个,在Spring 家族的其它框架中也有实现。数据结构
这里扯得有点远了,今天不是来说这些条件控制注解的用法的,只是我发现了一个使用条件注解@ConditionalOnProperty
没法解决的问题。app
条件注入参考往期:Spring Boot 2 实战:使用 @Condition 注解来根据条件注入 Bean框架
下面是一段配置文件:ide
app: v1: foo: name: felord.cn description: 码农小胖哥 bar: name: ooxx.cn description: xxxxxx
对应配置类:学习
@Data @ConfigurationProperties("app") public class AppProperties { /** * */ private Map<String, V1> v1 = new HashMap<>(); /** * * * @author felord.cn * @since 1.0.0.RELEASE */ @Data public static class V1 { /** * name */ private String name; /** * description */ private String description; } }
特殊之处来了,yml
配置里的 foo
、bar
实际上是做为Map
中的key
来标识V1
的,和其它配置参数不一样,这个key
用户能够随意定义一个String
来标识,多是foo
,多是bar
,彻底根据开发者的喜爱进行主观定义。这个时候你想根据app.v1.*.name
(暂时用通配符*
)来进行@ConditionalOnProperty
判断是行不通的,由于你不肯定*
的值,该怎么办呢?ui
这里我花了一天的时间去摸索,最开始我认为Spring提供通配符(app.v1.*.name
)甚至是SpringEL
表达式能够拿到,可是搞了半天无功而返。code
忽然我想到以前看Spring Security OAuth2源码中有相似的逻辑。用过Spring Security OAuth2相关的都知道Spring Security OAuth2也要求用户自定义一个key
来标识本身的OAuth2客户端。好比我用Gitee的:
spring: security: oauth2: client: registration: gitee: client-id: xxxxxx client-secret: xxxxx
这里的
key
就是gitee
,固然这根据开发者心情决定,甚至你用zhangshan
做为key
均可以。
Spring Security OAuth2 提供了相关的条件注入思路,下面是其条件注入判断的核心类:
public class ClientsConfiguredCondition extends SpringBootCondition { private static final Bindable<Map<String, OAuth2ClientProperties.Registration>> STRING_REGISTRATION_MAP = Bindable .mapOf(String.class, OAuth2ClientProperties.Registration.class); @Override public ConditionOutcome getMatchOutcome(ConditionContext context, AnnotatedTypeMetadata metadata) { ConditionMessage.Builder message = ConditionMessage.forCondition("OAuth2 Clients Configured Condition"); Map<String, OAuth2ClientProperties.Registration> registrations = getRegistrations(context.getEnvironment()); if (!registrations.isEmpty()) { return ConditionOutcome.match(message.foundExactly("registered clients " + registrations.values().stream() .map(OAuth2ClientProperties.Registration::getClientId).collect(Collectors.joining(", ")))); } return ConditionOutcome.noMatch(message.notAvailable("registered clients")); } private Map<String, OAuth2ClientProperties.Registration> getRegistrations(Environment environment) { return Binder.get(environment).bind("spring.security.oauth2.client.registration", STRING_REGISTRATION_MAP) .orElse(Collections.emptyMap()); } }
显然OAuth2ClientProperties
的结构和咱们要验证的AppProperties
结构是同样的。因此上面的逻辑是能够抄过来的,它能够将环境配置中的带有不肯定key
的配置绑定到咱们的配置类AppProperties
中。核心的绑定逻辑是这一段:
Binder.get(environment).bind("spring.security.oauth2.client.registration", STRING_REGISTRATION_MAP)
首先经过Bindable
来声明一个可绑定的数据结构,这里调用了mapOf
方法声明了一个Map
的数据绑定结构。而后经过绑定的具体操做对象Binder
从配置环境接口Environment
中提取了spring.security.oauth2.client.registration
开头的配置属性并注入到Map
中去。既然咱们可以获取到了Map
,根据什么策略判断就彻底掌握在咱们手中了。
Bindable
为Spring Boot 2.0提供的数据绑定新特性,有兴趣可从spring.io获取更多信息。
接下来不用我说了吧,照葫芦画瓢还有谁不会呢?配合@Conditional
注解就能实现根据app.v1
下参数的实际状况来动态的进行Bean注入。
今天利用Spring Boot 2.0的数据绑定特性解决了一个实际需求,花了很多时间。当咱们解决问题陷入困境时,首先要去想一想有没有相似场景以及对应的解决方案。这一样说明平时的积累很重要,不少粉丝的问题其实公众号都有讲过,因此到处留心解释学问。多多留意:码农小胖哥,共同窗习,共同进步。
关注公众号:Felordcn 获取更多资讯