通过条件注解@Conditional细粒度的选择bean实例

在进行spring进行开发时,当某个接口有多种实现方式并且我们只想让一种生效时,比如声明如下一个接口和两个实现:

public interface LanggageService {
      String say();
}
public class ChineseServiceImpl implements LanggageService {
    @Override
    public String say() {
        return "你好";
    }
}
public class EnglishServiceImpl implements LanggageService {
    @Override
    public String say() {
        return "hello";
    }
}

我们通常使用xml为某个接口配置实现类;

    <bean id="languageService" class="com.liam.service.ChineseServiceImpl"></bean>

当然后来我们习惯使用注解的方式了,比如

   @Bean
    public LanggageService englishService(){
        return new EnglishServiceImpl();
     }

当然也可以通过@Component进行扫描注入,这也是为了去除java令人发指的诸多xml配置的一项进步,所以当使用spring boot进行开发时,我们尽量摒弃了诸多配置,不管是Hibernate,JPA亦或者是mybatis...

但是也显然,有时候我们需要通过配置来选择我们使用的bean,基于xml还好,但是基于注解的话通过重新编译打包显然是不能接受的。。比如测试环境我们设置的数据源是mysql,生产环境换成oracle了,当然我们通常使用的是profile进行配置,但还是不够灵活,我们需要更细粒度的管理bean..所以也就有了Conditional。。。(废话连篇。。。)

比如我们在spring boot中要通过application.properties中进行配置来选择我们使用的bean

say.method=chinese

则可以声明两个Condition

public class SayChineseCondition implements Condition {
    @Override
    public boolean matches(ConditionContext conditionContext, AnnotatedTypeMetadata annotatedTypeMetadata) {
      String m=  conditionContext.getEnvironment().getProperty("say.method").toString();
        boolean isChi= "chinese".equals(m);
        System.err.println("current chinese   "+ isChi);
        return isChi;
    }
}
public class SayEnglishCondition implements Condition {
    @Override
    public boolean matches(ConditionContext conditionContext, AnnotatedTypeMetadata annotatedTypeMetadata) {
      String m=  conditionContext.getEnvironment().getProperty("say.method").toString();
        boolean isEn= "english"==m;
        System.err.println("current english   "+ isEn);
        return isEn;
    }
}

然后实现类中配置上对应的Conditional:

@Service
@Conditional(SayChineseCondition.class)
public class ChineseServiceImpl implements LanggageService {
    @Override
    public String say() {
        return "你好";
    }
}

简单,灵活。

此处需要注意的是application.properties在spring boot中属于环境变量级别的配置加载,所以优先级较高,假如通过其他配置文件进行加载时借助PropertySource或者ConfigurationProperties之类的是不能如愿的,因为加载Condition时,我们的配置bean尚未加载,比如下面的方式是无效的。

@PropertySource(value = "first.properties")
public class SayEnglishCondition implements Condition {
    @Value("${say.method}")
    private String lang;
    public String getLang() {
        return lang;
    }
    public void setLang(String lang) {
        this.lang = lang;
    }
    @Override
    public boolean matches(ConditionContext conditionContext, AnnotatedTypeMetadata annotatedTypeMetadata) {
        String m=lang;//  conditionContext.getEnvironment().getProperty("say.method").toString();
        boolean isEn= "english".equals(m);
        System.err.println("current english   "+ isEn);
        return isEn;
    }
}

所以。。我们只能手动加载了。。

public class SayEnglishCondition implements Condition { 
    @Override
    public boolean matches(ConditionContext conditionContext, AnnotatedTypeMetadata annotatedTypeMetadata) {
        Properties properties = new Properties();
        try {
            properties.load(conditionContext.getResourceLoader().getResource("first.properties").getInputStream());
        } catch (IOException ex) {
            ex.printStackTrace();
        }
        boolean isEn = "english".equals(properties.getProperty("say.method"));
        System.err.println("current english   " + isEn);
        return isEn;
    }
}

 

posted @ 2017-12-16 22:42  liamyu  阅读(2269)  评论(0编辑  收藏  举报