最后
每年转战互联网行业的人很多,说白了也是冲着高薪去的,不管你是即将步入这个行业还是想转行,学习是必不可少的。作为一个Java开发,学习成了日常生活的一部分,不学习你就会被这个行业淘汰,这也是这个行业残酷的现实。
如果你对Java感兴趣,想要转行改变自己,那就要趁着机遇行动起来。或许,这份限量版的Java零基础宝典能够对你有所帮助。
最近有个读者在面试,面试中被问到了这样一个问题“看你项目中用到了springboot
,你说下springboot
的自动配置是怎么实现的?”这应该是一个springboot
里面最最常见的一个面试题了。下面我们就来带着这个问题一起解剖下springBoot
的自动配置原理吧。
SpringMvc和SpringBoot对比
首先我们回顾下原来搭建一个springmvc
的hello-word
的web
项目(xml
配置的)我们是不是要在pom
中导入各种依赖,然后各个依赖有可能还会存在版本冲突需要各种排除。当你历尽千辛万苦的把依赖解决了,然后还需要编写web.xml、springmvc.xml
配置文件等。我们只想写个hello-word
项目而已,确把一大把的时间都花在了配置文件和jar
包的依赖上面。大大的影响了我们开发的效率,以及加大了web
开发的难度。为了简化这复杂的配置、以及各个版本的冲突依赖关系,springBoot
就应运而生。我们现在通过idea
创建一个springboot
项目只要分分钟就解决了,你不需要关心各种配置(基本实现零配置)。让你真正的实现了开箱即用。SpringBoot
帮你节约了大量的时间去陪女朋友,不对程序员怎么会有女朋友呢?(没有的话也是可以new一个的)它的出现不仅可以让你把更多的时间都花在你的业务逻辑开发上,而且还大大的降低了web
开发的门槛。所以SpringBoot
还是比较善解人衣的,错啦错啦是善解人意,知道开发人员的痛点在哪。
SpringBoot自动配置加载
既然Springboot
尽管这么好用,但是作为一个使用者,我们还是比较好奇它是怎么帮我们实现开箱即用的。Spring Boot
有一个全局配置文件:application.properties或application.yml
。在这个全局文件里面可以配置各种各样的参数比如你想改个端口啦server.port
或者想调整下日志的级别啦通通都可以配置。更多其他可以配置的属性可以参照官网。https://docs.spring.io/spring-boot/docs/2.3.0.RELEASE/reference/htmlsingle/#common-application-properties
这么多属性,这些属性在项目是怎么起作用的呢?SpringBoot
项目看下来啥配置也没有,配置”(application.properties或application.yml
除外),既 然从配置上面找不到突破口,那么我们就只能从启动类上面找入口了。启动类也就一个光秃秃的一个main
方法,类上面仅有一个注SpringBootApplication
这个注解是Spring Boot
项目必不可少的注解。那么自动配置原理一定和这个注解有着千丝万缕的联系!我们下面来一起看看这个注解吧。 @SpringBootApplication注解
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@SpringBootConfiguration
@EnableAutoConfiguration
@ComponentScan(excludeFilters = { @Filter(type = FilterType.CUSTOM, classes = TypeExcludeFilter.class),
@Filter(type = FilterType.CUSTOM, classes = AutoConfigurationExcludeFilter.class) })
public @interface SpringBootApplication {
这里最上面四个注解的话没啥好说的,基本上自己实现过自定义注解的话,都知道分别是什么意思。 - @SpringBootConfiguration
继承自@Configuration
,二者功能也一致,标注当前类是配置类。 - @ComponentScan
用于类或接口上主要是指定扫描路径,跟Xml里面的<context:component-scan base-package="" />
配置一样。springboot
如果不写这个扫描路径的话,默认就是启动类的路径。 - @EnableAutoConfiguration
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@AutoConfigurationPackage
@Import(AutoConfigurationImportSelector.class)
public @interface EnableAutoConfiguration {
这个注解我们重点看下AutoConfigurationImportSelector
这个类getCandidateConfigurations
这个方法里面通过SpringFactoriesLoader.loadFactoryNames()
扫描所有具有META-INF/spring.factories
的jar
包( spring.factories 我们可以理解成 Spring Boot
自己的 SPI
机制)。 spring-boot-autoconfigure-x.x.x.x.jar
里就有一个spring.factories文件。spring.factories
文件由一组一组的Key = value
的形式,其中一个key
是EnableAutoConfiguration类的全类名,而它的value是一个以AutoConfiguration
结尾的类名的列表,有redis、mq
等这些类名以逗号分隔。
我们在回到getAutoConfigurationEntry
这个方法当执行完getCandidateConfigurations
这个方法的时候我们可以看到此时总共加载了127
个自动配置类。
这些类难道都要加载进去吗?springboot
还是没有那么傻的,它提倡的话是按需加载。 - 它会去掉重复的类 - 过滤掉我们配置了exclude
注解的类下面配置就会过滤掉RestTemplateAutoConfiguration
这个类
- 经过上面的处理,剩下的这写自动配置的类如果要起作用的话,是需要满足一定的条件的。这些条件的满足的话spring boot
是通过条件注解来实现的。
@ConditionalOnBean:当容器里有指定Bean的条件下 @ConditionalOnClass:当类路径下有指定的类的条件下 @ConditionalOnExpression:基于SpEL表达式为true的时候作为判断条件才去实例化 @ConditionalOnJava:基于JVM版本作为判断条件 @ConditionalOnJndi:在JNDI存在的条件下查找指定的位置 @ConditionalOnMissingBean:当容器里没有指定Bean的情况下 @ConditionalOnMissingClass:当容器里没有指定类的情况下 @ConditionalOnWebApplication:当前项目时Web项目的条件下 @ConditionalOnNotWebApplication:当前项目不是Web项目的条件下 @ConditionalOnProperty:指定的属性是否有指定的值 @ConditionalOnResource:类路径是否有指定的值 @ConditionalOnOnSingleCandidate:当指定Bean在容器中只有一个,或者有多个但是指定首选的Bean
这些注解都组合了@Conditional
注解,只是使用了不同的条件组合最后为true时才会去实例化需要实例化的类,否则忽略过滤掉。我们在回到代码可以看到经过了条件判断过滤后我们剩下符合条件的自动配置类只剩23个了。其他的都是因为不满足条件注解而被过滤了。
如果我们想知道哪些自动配置类被过滤了,是由于什么原因被过滤了,以及加载了哪些类等。spring boot
都为我们记录了日志。还是非常贴心的。我们可以调整下我们日志的级别改为debug
。然后我们就能看到以下日志了
这里就截取了部分日志。总共分别有下面四部分日志: - Positive matches
:@Conditional
条件为真,配置类被Spring容器加载。 - Negative matches:
@Conditional
条件为假,配置类未被Spring容器加载。 - Exclusions
: 我们明确了不需要加载的类。比如在上面启动类配置的RestTemplateAutoConfiguration
类 - Unconditional classes
: 自动配置类不包含任何类级别的条件,也就是说,类始终会被自动加载。
自动配置生效
惊喜
最后还准备了一套上面资料对应的面试题(有答案哦)和面试时的高频面试算法题(如果面试准备时间不够,那么集中把这些算法题做完即可,命中率高达85%+)
视频+实战项目源码】](https://bbs.csdn.net/forums/4f45ff00ff254613a03fab5e56a57acb)收录**