Spring Boot自定义一个starter
日常工作中对于Spring Boot
提供的一些启动器可能已经足够使用了,但是不可避免的需要自定义启动器,比如整合一个陌生的组件,也想要达到开箱即用的效果。
这篇文章将会介绍如何自定义一个启动器,同时对于自动配置类的执行顺序做一个详细的分析。
如何自定义一个starter?
启动器的核心其实就是自动配置类,在自动配置源码分析的章节已经介绍过,AutoConfigurationImportSelector
是从spring.factories
中加载自动配置类,因此只需要将自定义的自动配置类设置在该文件中即可。
读过源码的朋友都知道自动配置类常用的一些注解,总结如下:
@Configuration
:该注解标志这是一个配置类,「自动配置类可以不加该注解」。@EnableConfigurationProperties
:这个配置也是经常使用了,使得指定的属性配置生效。一般自动配置类都需要从全局属性配置中读取自定义的配置,这就是一个开关。@ConditionalOnXxxx
:该注解是自动配置类的核心了,自动配置类既要启动时自动配置,又要保证用户用户自定义的配置覆盖掉自动配置,该注解就是一个条件语句,只有当指定条件成立才会执行某操作。@AutoConfigureAfter
:指定自动配置类的执行先后顺序,下文详细介绍。@AutoConfigureBefore
:指定自动配置列的执行先后顺序,下文详细介绍。@AutoConfigureOrder
:指定自动配置类的优先级,下文详细介绍。
有了以上准备,自定义一个starter
非常简单,分为两个步骤。
1. 准备自己的自动配置类
启动器的灵魂核心就是自动配置类,因此需要首先创建一个自动配置类,如下:
1 | @Configuration |
以上自动配置类只是作者简单的按照格式随手写了一个,真实开发中需要根据启动器的业务做默认配置。
2. 将自动配置类设置在spring.factories
标注了@Configuration
注解的自动配置类如果不放在spring.factories
文件中,仅仅是一个普通的配置类而已。想要其成为自动配置类,需要在spring.factories
文件中设置,如下:
1 | # Auto Configure |
经过以上的配置,粗略的启动器完成了,只需要打包,然后Maven
引入即可工作。
如何指定自动配置类的执行顺序?
自动配置类需要定义执行顺序吗?
答案:肯定的。比如Mybatis
的自动配置类,肯定要在数据源的自动配置类之后执行,否则如何创建SqlSessionFactory
?
如何自定义自动配置类的执行顺序呢?此时就需要用到上文提到的三个注解,如下:
@AutoConfigureAfter
:当前配置类在指定配置类之后执行@AutoConfigureBefore
:当前配置类在指定配置类之前执行@AutoConfigureOrder
:指定优先级,数值越小,优先级越高。
分享一个经典的误区
对于Spring Boot
不是很了解的人写出的代码真是「不堪入目」,曾经看过有人在「普通的配置类」上使用@AutoConfigurexxx
注解,如下;
1 | @Configuration |
是不是感觉很爽,原来还能这么指定配置类的执行顺序…..(此处省略一万字)
可能有时候走了狗屎运给你一种错觉还真的配置成功了。实际上这种方式是不可行的,以上三个注解只有针对自动配置类才会生效。
源码分析自动配置类如何排序?
其实关键的代码还是在AutoConfigurationImportSelector
中,将自动配置类从spring.factories
加载出来之后会根据条件排序,在selectImports()
方法中最后一行代码如下:
1 | return sortAutoConfigurations(processedConfigurations, getAutoConfigurationMetadata()).stream() |
上面的代码则是将排序好的自动配置类返回,跟进代码,发现最终的实现都在AutoConfigurationSorter.getInPriorityOrder()
方法中,逻辑如下:
1 | List<String> getInPriorityOrder(Collection<String> classNames) { |
排序也是按照先后顺序,如下:
- 先按照字母排序
- 按照
@AutoConfigureOrder
优先级排序 - 最终按照
@AutoConfigureAfter
、@AutoConfigureBefore
排序
从上面配置的顺序可以知道,最终决定权还是在
@AutoConfigureAfter
、@AutoConfigureBefore
这两个注解。