
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{.....}