(二)spring 高级装配-Condition -条件化的bean

时间:2023-03-09 16:40:35
(二)spring 高级装配-Condition -条件化的bean

Condition:满足某个特定条件的情况下创建bean

条件化配置bean:

a:@Conditional 指定一个class ,它指明了通过条件对比的类。如果没有指定class则通过Conditon接口进行条件对比:

b:@Conditional 指定的类可以是任意实现了Condition接口的类

c:指定的类需要重写matches方法

1.例子1:

@Bean
@Conditional(MagicExistsConditon.class) //条件化的创建bean
public MagicBean magicBean(){
return new MagicBean();
}

Condition接口:

public interface Condition{
boolean matches(ConditionContext context,AnnotatedTypeMetadata metadata);
}

MagicExistsConditon实现类:

 public class MagicExistsCondition implements Condition{
public boolean matches(ConditionContext context,
AnnotatedTypeMetadata metadata)
Environment env = context.getEnvironment();
return env.containsProperty("magic");
}
}

例子1,Conditional注解指定了实现类作为条件,如果环境中包含magic属性,则会创建magicBean;

ConditionContext 是一个接口:

 public interface ConditionContext{
BeanDefinitionRegistry getRegistry();
ConfigurableListableBeanFactory getBeanFactory();
Environment getEnvironment();
ResourceLoader getResourceLoader();
ClassLoader getClassLoader();
}

通过ConditionContext可以检查bean的定义;

通过BeanFactory,可以检查bean是否存在,甚至探查bean的属性;

检查环境变量是否存在以及它的值是什么;

ResourceLoader所加载 的资源;

ClassLoader加载并检查类是否存在;

AnnotatedTypeMetadata接口:能够让我们检查带有@Bean注解的方法上还有什么其他的注解;

publlic interface AnnotatedTypeMetadata{
boolean isAnnotated(String annotationType);
Map<String,Object> getAnnotationAttributes(String annotationType);
Map<String,Object> getAnnotationAttributes(String annotationType,boolean classValuesAsString);
MultiValueMap<String,Object> getAllAnnotationAttributes(
String annotationType);
MultiValueMap<String,Object> getAllAnnotationAttributes(String annotationType,boolean classValuesAsString);
}

a:判断是否还有其他注解;

b:检查@Bean注解的方法上其他注解的属性;

例子2:@Profile注解

@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.Type,ElementType.METHOD})
@Documented
@Conditional(ProfileCondition.class)
public @interface Profile{
String[] value();
}

Profile注解使用了@Conditional注解。

通过查看ProfileConditon的源码,可以发现,它借助ConditonContext得到Environment来检查该Profile是否处于激活状态。

public boolean matches(ConditionContext contex,AnnotatedTypeMeatadata metadata){
if(context.getEnvironment() != null){
MultiValueMap<String,Object> attrs = metadata.getAllAnnotationAttributes(Profile.class.getName());
if(atts != null){
for(Object value: attrs.get("value")){
if(context.getEnvironment().acceptsProfiles((String[]) value)){
return true;
}
return false;
}
}
return true;
}
}

例子3:处理自动装配的歧义性:

当通过@Autowared注入一个接口的时候,如果该接口有多个实现类的时候,spring,不知道该注入那个实现类的代理。

解决方法:

a:通过在实现类上添加注解@Primary与@Component同时使用,代表当出现多个的时候,此类优先注入。

b:存在多个首选的时候,通过@Qualifier注解,@Qualifier(“iceCream”),value的值代表想要注入的bean的ID

c:出现多个ID相同的时候,创建自定义限定符,即@Qualifier(“iceCream”),自定义value的值,如下,cold即自定义的限定符

@Component
@Qualifier("cold")
public class IceCream implements Dersert{......}

d:当出现重复的自定义符时,通过自定义限定符注解,自定义一个注解,加上@Qualifier注解,则自定义的注解也有了Qualifier的功能。当出现自定义限定符重复的时候,可以通过添加多个自定义限定符注解,实现要注入的是哪个bean

@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.Constructor,ElementType.FILED,ElementType.METHOD,ElementType.TYPE})
@Qualifier
public @interface cold{.....}
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.Constructor,ElementType.FILED,ElementType.METHOD,ElementType.TYPE})
@Qualifier
public @interface creamy{.....}