
公众号高质量博文整理(请戳我)
-
SpringBoot 源码常用注解拾遗
-
SpringBoot 启动过程
-
SpringBoot 自动配置原理
1. SpringBoot 源码常用注解拾遗
组合注解
-
元注解:可以注解到别的注解上的注解。
-
组合注解:被注解的注解我们就称之为组合注解。
@Value
public class Person {private String name;}
<bean class="Person"><property name ="name" value="i am name"></property></bean>
-
字面量
-
通过
${key}
方式从环境变量中获取值
-
通过
${key}
方式全局配置文件中获取值
-
#{SpEL}
@Value(${key})的方式获取全局配置文件中的指定配置项。微信搜索 web_resource 回复 爆文 送你高质量技术博文。
@ConfigurationProperties
@Value的方式去配置项需要一个一个去取,这就显得有点 low 了。我们可以使用
@ConfigurationProperties。
@ConfigurationProperties的类的所有属性和配置文件中相关的配置项进行绑定。(默认从全局配置文件中获取配置值),绑定之后我们就可以通过这个类去访问全局配置文件中的属性值了。
person.name=kundyperson.age=13person.sex=male
(prefix = "person")public class Person {private String name;private Integer age;private String sex;}
@ConfigurationProperties有一个 prefix 参数,主要是用来指定该配置项在配置文件中的前缀。
@Import
-
@Import 注解在 4.2 之前只支持导入配置类。
-
在4.2之后 @Import 注解支持导入普通的 java 类,并将其声明成一个 bean。
-
直接导入普通的 Java 类。
-
配合自定义的 ImportSelector 使用。
-
配合 ImportBeanDefinitionRegistrar 使用。
1. 直接导入普通的 Java 类
public class Circle {public void sayHi() {System.out.println("Circle sayHi()");}}
public class MainConfig {}
public static void main(String[] args) {ApplicationContext context = new AnnotationConfigApplicationContext(MainConfig.class);Circle circle = context.getBean(Circle.class);circle.sayHi();}
Circle sayHi()
2. 配合自定义的 ImportSelector 使用
public class Triangle {public void sayHi(){System.out.println("Triangle sayHi()");}}
public class MyImportSelector implements ImportSelector {public String[] selectImports(AnnotationMetadata annotationMetadata) {return new String[]{"annotation.importannotation.waytwo.Triangle"};}}
@Import({Circle.class,MyImportSelector.class})@Configurationpublic class MainConfigTwo {}
public static void main(String[] args) {ApplicationContext context = new AnnotationConfigApplicationContext(MainConfigTwo.class);Circle circle = context.getBean(Circle.class);Triangle triangle = context.getBean(Triangle.class);circle.sayHi();triangle.sayHi();}
Circle sayHi()
Triangle sayHi()
3. 配合 ImportBeanDefinitionRegistrar 使用
public class Rectangle {public void sayHi() {System.out.println("Rectangle sayHi()");}}
public class MyImportBeanDefinitionRegistrar implements ImportBeanDefinitionRegistrar {public void registerBeanDefinitions(AnnotationMetadata annotationMetadata, BeanDefinitionRegistry beanDefinitionRegistry) {RootBeanDefinition rootBeanDefinition = new RootBeanDefinition(Rectangle.class);// 注册一个名字叫做 rectangle 的 beanbeanDefinitionRegistry.registerBeanDefinition("rectangle", rootBeanDefinition);}}
@Import({Circle.class, MyImportSelector.class, MyImportBeanDefinitionRegistrar.class})@Configurationpublic class MainConfigThree {}
public static void main(String[] args) {ApplicationContext context = new AnnotationConfigApplicationContext(MainConfigThree.class);Circle circle = context.getBean(Circle.class);Triangle triangle = context.getBean(Triangle.class);Rectangle rectangle = context.getBean(Rectangle.class);circle.sayHi();triangle.sayHi();rectangle.sayHi();}
Circle sayHi()
Triangle sayHi()
Rectangle sayHi()
@Conditional
@Conditional 注释可以实现只有在特定条件满足时才启用一些配置。
public class ConditionBean {public void sayHi() {System.out.println("ConditionBean sayHi()");}}
public class MyCondition implements Condition {public boolean matches(ConditionContext conditionContext, AnnotatedTypeMetadata annotatedTypeMetadata) {return true;}}
(MyCondition.class)public class ConditionConfig {public ConditionBean conditionBean(){return new ConditionBean();}}
public static void main(String[] args) {ApplicationContext context = new AnnotationConfigApplicationContext(ConditionConfig.class);ConditionBean conditionBean = context.getBean(ConditionBean.class);conditionBean.sayHi();}

2. SpringBoot 启动过程
-
ApplicationContextInitializer
-
ApplicationRunner
-
CommandLineRunner
-
SpringApplicationRunListener
SpringApplication.run() -> run(new Class[]{primarySource}, args) -> new SpringApplication(primarySources)).run(args)。
new SpringApplication(primarySources)).run(args)这个方法。
-
创建 SpringApplication 对象。
-
运行 run() 方法。
创建 SpringApplication 对象
public SpringApplication(ResourceLoader resourceLoader, Class... primarySources) {this.sources = new LinkedHashSet();this.bannerMode = Mode.CONSOLE;this.logStartupInfo = true;this.addCommandLineProperties = true;this.addConversionService = true;this.headless = true;this.registerShutdownHook = true;this.additionalProfiles = new HashSet();this.isCustomEnvironment = false;this.resourceLoader = resourceLoader;Assert.notNull(primarySources, "PrimarySources must not be null");// 保存主配置类(这里是一个数组,说明可以有多个主配置类)this.primarySources = new LinkedHashSet(Arrays.asList(primarySources));// 判断当前是否是一个 Web 应用this.webApplicationType = WebApplicationType.deduceFromClasspath();// 从类路径下找到 META/INF/Spring.factories 配置的所有 ApplicationContextInitializer,然后保存起来this.setInitializers(this.getSpringFactoriesInstances(ApplicationContextInitializer.class));// 从类路径下找到 META/INF/Spring.factories 配置的所有 ApplicationListener,然后保存起来this.setListeners(this.getSpringFactoriesInstances(ApplicationListener.class));// 从多个配置类中找到有 main 方法的主配置类(只有一个)this.mainApplicationClass = this.deduceMainApplicationClass();}
运行 run() 方法
public ConfigurableApplicationContext run(String... args) {// 创建计时器StopWatch stopWatch = new StopWatch();stopWatch.start();// 声明 IOC 容器ConfigurableApplicationContext context = null;Collection<SpringBootExceptionReporter> exceptionReporters = new ArrayList();this.configureHeadlessProperty();// 从类路径下找到 META/INF/Spring.factories 获取 SpringApplicationRunListenersSpringApplicationRunListeners listeners = this.getRunListeners(args);// 回调所有 SpringApplicationRunListeners 的 starting() 方法listeners.starting();Collection exceptionReporters;try {// 封装命令行参数ApplicationArguments applicationArguments = new DefaultApplicationArguments(args);// 准备环境,包括创建环境,创建环境完成后回调 SpringApplicationRunListeners#environmentPrepared()方法,表示环境准备完成ConfigurableEnvironment environment = this.prepareEnvironment(listeners, applicationArguments);this.configureIgnoreBeanInfo(environment);// 打印 BannerBanner printedBanner = this.printBanner(environment);// 创建 IOC 容器(决定创建 web 的 IOC 容器还是普通的 IOC 容器)context = this.createApplicationContext();exceptionReporters = this.getSpringFactoriesInstances(SpringBootExceptionReporter.class, new Class[]{ConfigurableApplicationContext.class}, context);/** 准备上下文环境,将 environment 保存到 IOC 容器中,并且调用 applyInitializers() 方法* applyInitializers() 方法回调之前保存的所有的 ApplicationContextInitializer 的 initialize() 方法* 然后回调所有的 SpringApplicationRunListener#contextPrepared() 方法* 最后回调所有的 SpringApplicationRunListener#contextLoaded() 方法*/this.prepareContext(context, environment, listeners, applicationArguments, printedBanner);// 刷新容器,IOC 容器初始化(如果是 Web 应用还会创建嵌入式的 Tomcat),扫描、创建、加载所有组件的地方this.refreshContext(context);// 从 IOC 容器中获取所有的 ApplicationRunner 和 CommandLineRunner 进行回调this.afterRefresh(context, applicationArguments);stopWatch.stop();if (this.logStartupInfo) {(new StartupInfoLogger(this.mainApplicationClass)).logStarted(this.getApplicationLog(), stopWatch);}// 调用 所有 SpringApplicationRunListeners#started()方法listeners.started(context);this.callRunners(context, applicationArguments);} catch (Throwable var10) {this.handleRunFailure(context, var10, exceptionReporters, listeners);throw new IllegalStateException(var10);}try {listeners.running(context);return context;} catch (Throwable var9) {this.handleRunFailure(context, var9, exceptionReporters, (SpringApplicationRunListeners)null);throw new IllegalStateException(var9);}}
小结:
3. SpringBoot 自动配置原理
@SpringBootApplication 注解
-
这个类是 SpringBoot 的主配置类。
-
SpringBoot 就应该运行这个类的 main 方法来启动 SpringBoot 应用。
@SpringBootConfiguration@EnableAutoConfiguration@ComponentScan(excludeFilters = {@Filter(type = FilterType.CUSTOM,classes = {TypeExcludeFilter.class}), @Filter(type = FilterType.CUSTOM,classes = {AutoConfigurationExcludeFilter.class})})public @interface SpringBootApplication {}
-
@SpringBootConfiguration:该注解表示这是一个 SpringBoot 的配置类,其实它就是一个 @Configuration 注解而已。
-
@ComponentScan:开启组件扫描。
-
@EnableAutoConfiguration:从名字就可以看出来,就是这个类开启自动配置的。嗯,自动配置的奥秘全都在这个注解里面。
@EnableAutoConfiguration 注解
@AutoConfigurationPackage@Import({AutoConfigurationImportSelector.class})public @interface EnableAutoConfiguration {
@Import({Registrar.class}),导入了一个 Registrar 的组件。关于 @Import 的用法文章上面也有介绍哦。

@Import({AutoConfigurationImportSelector.class})

getAutoConfigurationEntry() -> getCandidateConfigurations() -> loadFactoryNames()。
loadFactoryNames()方法传入了 EnableAutoConfiguration.class 这个参数。先记住这个参数,等下会用到。

-
从当前项目的类路径中获取所有 META-INF/spring.factories 这个文件下的信息。
-
将上面获取到的信息封装成一个 Map 返回。
-
从返回的 Map 中通过刚才传入的 EnableAutoConfiguration.class 参数,获取该 key 下的所有值。

META-INF/spring.factories这类文件是什么就不懵了。当然在很多第三方依赖中都会有这个文件,一般每导入一个第三方的依赖,除了本身的jar包以外,还会有一个 xxx-spring-boot-autoConfigure,这个就是第三方依赖自己编写的自动配置类。我们现在就以 spring-boot-autocongigure 这个依赖来说。微信搜索 web_resource 回复 爆文 送你高质量技术博文。

可以看到 EnableAutoConfiguration 下面有很多类,这些就是我们项目进行自动配置的类。
一句话:将类路径下 META-INF/spring.factories 里面配置的所有 EnableAutoConfiguration 的值加入到 Spring 容器中。
HttpEncodingAutoConfiguration
通过上面方式,所有的自动配置类就被导进主配置类中了。但是这么多的配置类,明显有很多自动配置我们平常是没有使用到的,没理由全部都生效吧。
接下来我们以 HttpEncodingAutoConfiguration为例来看一个自动配置类是怎么工作的。为啥选这个类呢?主要是这个类比较的简单典型。
先看一下该类标有的注解:
@ConditionalOnWebApplication(type = Type.SERVLET)@ConditionalOnProperty(prefix = ,value = {},matchIfMissing = true)public class HttpEncodingAutoConfiguration {}
-
@Configuration:标记为配置类。
-
@ConditionalOnWebApplication:web应用下才生效。
-
@ConditionalOnClass:指定的类(依赖)存在才生效。
-
@ConditionalOnProperty:主配置文件中存在指定的属性才生效。
-
@EnableConfigurationProperties({HttpProperties.class}):启动指定类的ConfigurationProperties功能;将配置文件中对应的值和 HttpProperties 绑定起来;并把 HttpProperties 加入到 IOC 容器中。
@EnableConfigurationProperties({HttpProperties.class})把配置文件中的配置项与当前 HttpProperties 类绑定上了。
@ConfigurationProperties(prefix =)// 从配置文件中获取指定的值和bean的属性进行绑定public class HttpProperties {}
小结:
-
SpringBoot启动会加载大量的自动配置类。
-
我们看需要的功能有没有SpringBoot默认写好的自动配置类。
-
我们再来看这个自动配置类中到底配置了那些组件(只要我们要用的组件有,我们就不需要再来配置了)。
-
给容器中自动配置类添加组件的时候,会从properties类中获取某些属性。我们就可以在配置文件中指定这些属性的值。
xxxAutoConfiguration:自动配置类给容器中添加组件。
xxxProperties:封装配置文件中相关属性。
不知道小伙伴们有没有发现,很多需要待加载的类都放在类路径下的META-INF/Spring.factories 文件下,而不是直接写死这代码中,这样做就可以很方便我们自己或者是第三方去扩展,我们也可以实现自己 starter,让SpringBoot 去加载。现在明白为什么 SpringBoot 可以实现零配置,开箱即用了吧!

荐
阅
读
在浏览器输入 URL 回车之后发生了什么?
接私活必备的 10 个开源项目

欢
文
章
,
点
个
在看

本文分享自微信公众号 - Java后端(web_resource)。
如有侵权,请联系 support@oschina.cn 删除。
本文参与“OSC源创计划”,欢迎正在阅读的你也加入,一起分享。
