如何使用SpringBootCondition更自由地定義條件化配置
Conditional如何使用
@Conditional 是 SpringFramework 的功能, SpringBoot 在它的基礎(chǔ)上定義了 @ConditionalOnClass , @ConditionalOnProperty 的一系列的注解來實現(xiàn)更豐富的內(nèi)容。
定義一個自定義標簽
import com.example.conditional.MyConditional; import org.springframework.context.annotation.Conditional; import java.lang.annotation.*; @Target({ElementType.TYPE,ElementType.METHOD}) @Retention(RetentionPolicy.RUNTIME) @Documented @Conditional(MyConditional.class) public @interface MyConditionalIAnnotation { String key(); String value(); }
自定義Conditional
import com.example.conditional.interfaceI.MyConditionalIAnnotation; import org.springframework.boot.autoconfigure.condition.ConditionOutcome; import org.springframework.boot.autoconfigure.condition.SpringBootCondition; import org.springframework.context.annotation.ConditionContext; import org.springframework.context.annotation.Conditional; import org.springframework.core.type.AnnotatedTypeMetadata; import java.util.Map; public class MyConditional extends SpringBootCondition { @Override public ConditionOutcome getMatchOutcome(ConditionContext context, AnnotatedTypeMetadata metadata) { Map<String, Object> annotationAttributes = metadata.getAnnotationAttributes(MyConditionalIAnnotation.class.getName()); Object key = annotationAttributes.get("key");// Object value = annotationAttributes.get("value"); if(key == null || value == null){ return new ConditionOutcome(false, "error"); } //獲取environment中的值 String key1 = context.getEnvironment().getProperty(key.toString()); if (value.equals(key1)) { //如果environment中的值與指定的value一致,則返回true return new ConditionOutcome(true, "ok"); } return new ConditionOutcome(false, "error"); } }
config配置
import com.example.conditional.interfaceI.MyConditionalIAnnotation; import com.example.conditional.service.MyConditionalService; import org.apache.log4j.Logger; import org.springframework.boot.autoconfigure.EnableAutoConfiguration; import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.ComponentScan; import org.springframework.context.annotation.Configuration; @Configuration public class MyConditionalConfig { public static Logger logger=Logger.getLogger(MyConditionalService.class); /** * 判斷MyConditional 是否符合條件,是則運行MyConditionalService * @return */ @MyConditionalIAnnotation(key = "com.example.conditional", value = "lbl") @ConditionalOnClass(MyConditionalService.class) @Bean public MyConditionalService initMyConditionService() { logger.info("MyConditionalService已加載。"); return new MyConditionalService(); } }
配置文件:application.propeties
spring.application.name=gateway server.port=8084 #conditional 動態(tài)配置,判斷該值是否等于lbl,是則創(chuàng)建MyConditionalService實例 com.example.conditional=lbl #支持自定義aop spring.aop.auto=true
SpringBootCondition 定義條件化配置
1 條件化配置
Spring提供了多種實現(xiàn)化條件化配置的選擇,如ConditionalOnProperty和ConditionalOnClass等。
用法如下:
@ConditionalOnProperty(prefix = "pkslow", name = "service", havingValue = "larry")
還有:
@ConditionalOnBean(僅僅在當前上下文中存在某個對象時,才會實例化一個Bean) @ConditionalOnClass(某個class位于類路徑上,才會實例化一個Bean) @ConditionalOnExpression(當表達式為true的時候,才會實例化一個Bean) @ConditionalOnMissingBean(僅僅在當前上下文中不存在某個對象時,才會實例化一個Bean) @ConditionalOnMissingClass(某個class類路徑上不存在的時候,才會實例化一個Bean) @ConditionalOnNotWebApplication(不是web應(yīng)用)
但有時候我們需要更靈活的自定義條件配置,這時可以通過繼承SpringBootCondition類來實現(xiàn)。
2 繼承SpringBootCondition
自己根據(jù)需求實現(xiàn)自己的判斷邏輯,我的實現(xiàn)如下:
public class PkslowCondition extends SpringBootCondition { @Override public ConditionOutcome getMatchOutcome(ConditionContext context, AnnotatedTypeMetadata metadata) { BindResult<List<String>> maxBindResult = Binder.get(context.getEnvironment()).bind("pkslow.condition.max", Bindable.listOf(String.class)); BindResult<List<String>> minBindResult = Binder.get(context.getEnvironment()).bind("pkslow.condition.min", Bindable.listOf(String.class)); if ( (maxBindResult.isBound() && !maxBindResult.get().isEmpty()) && (minBindResult.isBound() && !minBindResult.get().isEmpty()) ) { List<String> maxs = maxBindResult.get(); List<String> mins = minBindResult.get(); int max = Integer.parseInt(maxs.get(0)); int min = Integer.parseInt(mins.get(0)); if (max < 1000 && min > 0) { return ConditionOutcome.match(); } } return ConditionOutcome.noMatch("pkslow.condition.max/pkslow.condition.min not matches"); } }
表示需要有配置屬性pkslow.condition.max/pkslow.condition.min才會生效,并且要求max<1000且min>0。
3 使用
完成自定義的條件類后,就可以使用它來限定一個配置類是否要生效了,使用如下:
@Conditional(PkslowCondition.class) @Configuration public class PkslowConfig { @PostConstruct public void postConstruct() { System.out.println("PkslowConfig called"); } }
4 總結(jié)
代碼請查看:https://github.com/LarryDpk/pkslow-samples
以上就是如何使用SpringBootCondition更自由地定義條件化配置的詳細內(nèi)容,更多關(guān)于SpringBootCondition 定義條件化配置的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Day14基礎(chǔ)不牢地動山搖-Java基礎(chǔ)
這篇文章主要給大家介紹了關(guān)于Java中方法使用的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧2021-08-08SpringBoot利用自定義json序列化器實現(xiàn)敏感字段數(shù)據(jù)脫敏詳解
這篇文章主要介紹了SpringBoot利用自定義json序列化器實現(xiàn)敏感字段數(shù)據(jù)脫敏詳解,因為案例代碼用到了hutool提供的DesensitizedUtil數(shù)據(jù)脫敏工具類,這里要引入hutool的依賴,如果你需要自定義 數(shù)據(jù)脫敏的邏輯,可以不引入這個依賴,需要的朋友可以參考下2024-01-01基于Java代碼實現(xiàn)游戲服務(wù)器生成全局唯一ID的方法匯總
我們在做服務(wù)器系統(tǒng)開發(fā)的時候,為了適應(yīng)數(shù)據(jù)大并發(fā)的請求,需要插入數(shù)據(jù)庫之前生成一個全局的唯一id,糾結(jié)全局唯一id怎么生成呢?下面小編給大家分享Java代碼實現(xiàn)游戲服務(wù)器生成全局唯一ID的方法匯總,涉及到優(yōu)劣勢方面的知識點,對此感興趣的朋友一起看看吧2016-10-10