Spring Boot自动配置,首先明白在创建springBoot时,会先导入一个parent坐标,里面包含一个springBoot项目会用到的所有基本坐标,这可能也是为啥每次创建项目时,要很久加载、下载东西,但是,虽然springBoot给你了父坐标,但是具体要用哪个,还得自己在下面的坐标处写一下,相当于声明一下。
我们了解Spring Boot的原理之前,得认识几个注解。
一.@Condition
这个注解就是帮助我们理解,SpringBoot底层的加载机制,在一个条件下,才会加载bean,对于SpringBoot来说就是,条件就是导入了某个坐标。
例1:在 Spring 的 IOC 容器中有一个 User 的 Bean,要求:导入Jedis坐标后,加载该Bean,没导入,则不加载。
静态判断:
写一个User实体类
public class User {
}
写一个类实现Condition类,重写里面的matches抽象方法,返回值是布尔类型的,就是用来条件约束的。用反射来判断是否存在Jedis坐标,也就是判断是否有Jedis的字节码文件。
matches 方法两个参数:
context:上下文对象,可以获取属性值,获取类加载器,获取BeanFactory等。
metadata:元数据对象,用于获取注解属性。
public class ClassCondition implements Condition {
@Override
public boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata) {
boolean flag = true;
try {
//判断redis.clients.jedis.Jedis.class文件是否存在
Class<?> aClass = Class.forName("redis.clients.jedis.Jedis");
} catch (ClassNotFoundException e) {
flag = false;
}
return flag;
}
}
在写一个配置类:@Bean注入,@Conditional,value属性是上面那个条件约束类的字节码。
@Configuration
public class UserConfig {
@Bean
@Conditional(value= ClassCondition.class)
public User student(){
return new User();
}
}
启动主配置文件,在添加了jedis坐标之后,可以获取到user的地址值,证明模拟spring boot成功。
此外还可以使用自定义注解@ConditionOnClass,将类的判断定义为动态的。
动态判断:
由于@ConditionOnClass注解的功能和@Conditional一样,我们看看@Conditional原码,直接复制就行。
//这是@Conditional
@Target({ElementType.TYPE, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface Conditional {
Class<? extends Condition>[] value();
}
//这是自定义的注解
@Target({ElementType.TYPE, ElementType.METHOD})//可以修饰在类与方法上
@Retention(RetentionPolicy.RUNTIME)//注解生效节点runtime
@Documented//生成文档
@Conditional(ClassCondition.class)
public @interface ConditionOnClass {
String[] value();//设置此注解的属性redis.clients.jedis.Jedis
}
而其中的@Conditional(ClassCondition.class)还是和静态的一致。
不同的是配置类中,用注解@ConditionOnClass,value值还是对应的反射类。
@Configuration
public class UserConfig {
@Bean
@ConditionOnClass(value={"com.alibaba.fastjson.JSON","redis.clients.jedis.Jedis"})
public User user(){
return new User();
}
启动类加载之后,如果有JSON和Jedis的坐标才会获取对象成功。
还有一些必要了解的有关配置的注解:
@Enable注解
SpringBoot中提供了很多Enable开头的注解,这些注解都是用于动态启用某些功能的。而其底层原理是使用@Import注 解导入一些配置类,实现Bean的动态加载。
@Import注解
@Enable底层依赖于@Import注解导入一些类,使用@Import导入的类会被Spring加载到IOC容器中。而@Import提供4中用法:
① 导入Bean
② 导入配置类
③ 导入 ImportSelector 实现类。一般用于加载配置文件中的类
④ 导入 ImportBeanDefinitionRegistrar 实现类。
@EnableAutoConfiguration 注解
主启动类:@SpringBootApplication注解
作用:SpringBoot的配置类 ,标注在某个类上 , 表示这是一个SpringBoot的配置类;
@SpringBootConfiguration
@EnableAutoConfiguration
@ComponentScan(
excludeFilters = {@Filter(
type = FilterType.CUSTOM,
classes = {TypeExcludeFilter.class}
), @Filter(
type = FilterType.CUSTOM,
classes = {AutoConfigurationExcludeFilter.class}
)}
)
public @interface SpringBootApplication {
// ......
}
扫描类:@ComponentScan
这个注解在Spring中很重要 ,它对应XML配置中的元素。
作用:自动扫描并加载符合条件的组件或者bean , 将这个bean定义加载到IOC容器中
@AutoConfigurationPackage :自动配置包
@EnableAutoConfiguration:开启自动配置功能
告诉SpringBoot开启自动配置功能,这样自动配置才能生效;此注解内部使用 @Import(AutoConfigurationImportSelector.class)来加载配置类。
@Import({AutoConfigurationImportSelector.class}) :给容器导入组件 ;
AutoConfigurationImportSelector :自动配置导入选择器,给容器中导入一些组件
配置文件位置:META-INF/spring.factories,该配置文件中定义了大量的配置类,当 SpringBoot 应用启动时,会自动加载这些配置类,初始化Bean
并不是所有的Bean都会被初始化,在配置类中使用Condition来加载满足条件的Bean