你一直在用的 Spring Boot Starters 到底是怎麼回事

Spring Boot 對比 Spring MVC 最大的優勢就是使用簡單,約定大於配置。不會像以前用 Spring MVC 的時候,時不時被 xml 配置文件搞的暈頭轉向,冷不防還由於 xml 配置上的一點疏忽,致使整個項目莫名其妙的不可用,頓感生活無所依戀,簡稱生無可戀。
這要歸功於組成了 Spring Boot 的各類各樣的 starters,有官方提供的,也有第三方開源出來。能夠這麼說,基本上你打算用的功能均可以找到,若是沒有找到,那就再找一找。
用 Spring Boot 的功能組件(例如 spring-boot-starter-actuator、 spring-boot-starter-data-redis 等)的步驟很是簡單,用著名的把大象放冰箱的方法來歸納的話,有如下三步就能夠完成組件功能的使用:
STEP 1
在 pom 文件中引入對應的包,例如:
<dependency> 
  <groupId>org.springframework.boot</groupId>  
  <artifactId>spring-boot-starter-actuator</artifactId> 
</dependency>複製代碼
STEP 2
在應用配置文件中加入相應的配置,配置都是組件約定好的,須要查看官方文檔或者相關說明。有些比較複雜的組件,對應的參數和規則也相應的較多,有點可能多大幾十上百了。
STEP 3
以上兩步都正常的狀況下,咱們就可使用組件提供的相關接口來開發業務功能了。
沒錯吧,這個過程咱們在平常的開發中不知道已經實踐了多少遍。那麼 Spring Boot 爲何能作到如此簡單易用呢,它內部是什麼樣的工做機制呢,不知道你有沒有研究過。
如下是爲了理解 Spring Boot 組件的實現機制而製做的一個 demo starter。理解其中的原理,對咱們往後的工做有什麼意義呢?
1. 遇到問題的時候,能夠幫助咱們更有頭緒的排查問題;
2. 能夠幫助咱們正確的閱讀源代碼,組件的切入口在哪兒,配置屬性是什麼等等;
以上
如下,開始實現這個簡單的 starter,這個 starter 並無什麼實際的功能,只是爲了作個演示而已。
開始以前,咱們要理解一下 spring boot starter 是什麼呢?
實際上 starter 並不會包含多少功能代碼,咱們能夠把它理解成一個「鏈接包」(我本身造的概念),按照這個概念來講: 它首先是一個包,一個集合,它把須要用的其餘功能組件囊括進來,放到本身的 pom 文件中。 而後它是一個鏈接,把它引入的組件和咱們的項目作一個鏈接,而且在中間幫咱們省去複雜的配置,力圖作到使用最簡單。
實現一個 starter 有四個要素:
  1. starter 命名 ;
  2. 自動配置類,用來初始化相關的 bean ;
  3. 指明自動配置類的配置文件 spring.factories ;
  4. 自定義屬性實體類,聲明 starter 的應用配置屬性 ;
好了,開始實現咱們的 demo
1. 給 starter 起個名字
也就是咱們使用它的時候在 pom 中引用的 artifactId。命名有有規則的,官方規定:
官方的 starter 的命名格式爲 spring-boot-starter-{name} ,例如上面提到的 spring-boot-starter-actuator。
非官方的 starter 的命名格式爲 {name}-spring-boot-starter,咱們把自定的 starter 命名爲 kite-spring-boot-starter,命名在 pom 文件裏。
<groupId>kite.springcloud</groupId>
<artifactId>kite-spring-boot-starter</artifactId>
<packaging>jar</packaging>
<version>1.0-SNAPSHOT</version>複製代碼
2. 引入自動配置包及其它相關依賴包
實現 starter 主要依賴自動配置註解,因此要在 pom 中引入自動配置相關的兩個 jar 包
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-configuration-processor</artifactId>
</dependency>
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-autoconfigure</artifactId>
</dependency>複製代碼
除此以外,依賴的其餘包固然也要引進來。
3. 建立 spring.factories 文件
在 resource/META-INF 目錄下建立名稱爲 spring.factories 的文件,爲何在這裏?當 Spring Boot 啓動的時候,會在 classpath 下尋找全部名稱爲 spring.factories 的文件,而後運行裏面的配置指定的自動加載類,將指定類(一個或多個)中的相關 bean 初始化。
例如本例中的配置信息是這樣的:
org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
  kite.springcloud.boot.starter.example.KiteAutoConfigure複製代碼
等號前面是固定的寫法,後面就是咱們自定義的自動配置類了,若是有多個的話,用英文逗號分隔開。
4. 編寫自動配置類
自動配置類是用來初始化 starter 中的相關 bean 的。能夠說是實現 starter 最核心的功能。
@Configuration
@ConditionalOnClass(KiteService.class)
@EnableConfigurationProperties(KiteProperties.class)
@Slf4j
public class KiteAutoConfigure {

    @Autowired
    private KiteProperties kiteProperties;

    @Bean
    @ConditionalOnMissingBean(KiteService.class)
    @ConditionalOnProperty(prefix = "kite.example",value = "enabled", havingValue = "true")
    KiteService kiteService(){
        return new KiteService(kiteProperties);
    }
}複製代碼
代碼很是簡單,放眼望去,最多的就是各類註解。
@Configuration 這個不用解釋,表示這是個自動配置類,咱們平時作項目時也會用到,通常是用做讀取配置文件的時候。
@ConditionalOnClass(KiteService.class) :
只有在 classpath 中找到 KiteService 類的狀況下,纔會解析此自動配置類,不然不解析。
@EnableConfigurationProperties(KiteProperties.class):
啓用配置類。
@Bean:實例化一個 bean 。
@ConditionalOnMissingBean(KiteService.class):
與 @Bean 配合使用,只有在當前上下文中不存在某個 bean 的狀況下才會執行所註解的代碼塊,也就是當前上下文尚未 KiteService 的 bean 實例的狀況下,纔會執行 kiteService() 方法,從而實例化一個 bean 實例出來。
@ConditionalOnProperty:
當應用配置文件中有相關的配置纔會執行其所註解的代碼塊。
這個類的總體含義就是: 當 classpath 中存在 KiteService 類時解析此配置類,什麼狀況下才會在 classpath 中存在呢,就是項目引用了相關的 jar 包。而且在上下文中沒有 KiteService 的 bean 實例的狀況下,new 一個實例出來,而且將應用配置中的相關配置值傳入。
5. 實現屬性配置類
@Data
@ConfigurationProperties("kite.example")
public class KiteProperties {

    private String host;

    private int port;
}複製代碼
配置類很簡單,只有兩個屬性,一個 host ,一個 port 。配置參數以 kite.example 做爲前綴。稍後咱們在使用這個 starter 的時候會看到如何聲明配置屬性。
6. 實現相關功能類
也就是前面一直提到的 KiteService,其實嚴格來說,這個業務功能類不該該放在 starter 中,應該放在單獨的 jar 包裏,可是此處 demo 很是簡單,也就在這裏寫了。
@Slf4j
public class KiteService {

    private String host;

    private int port;

    public KiteService(KiteProperties kiteProperties){
        this.host = kiteProperties.getHost();
        this.port = kiteProperties.getPort();
    }

    public void print(){
        log.info(this.host + ":" +this.port);
    }
}複製代碼
一個構造函數和一個 print 方法。
7. 打包
經過 maven 命令將這個 starter 安裝到本地 maven 倉庫
mvn install複製代碼
也能夠經過 mvn package deploy 發佈到你的私服
或者發佈到中央倉庫。
上面已經完成了 starter 的開發,並安裝到了本地倉庫,而後就是在咱們的項目中使用它了。
1. 建立項目,在 pom 中引用
<dependency>
    <groupId>kite.springcloud</groupId>
    <artifactId>kite-spring-boot-starter</artifactId>
    <version>1.0-SNAPSHOT</version>
</dependency>複製代碼
2. 應用配置項
建立 application.yml ,配置以下:
server:
  port: 3801
kite:
  example:
    enabled: true  # 開啓才生效
    host: 127.0.0.1
    port: 3801複製代碼
3. 調用 KiteService 的服務方法
@RestController
@RequestMapping(value = "use")
public class UseController {

    @Autowired
    private KiteService kiteService;

    @GetMapping(value = "print")
    public void print(){
        kiteService.print();
    }
}複製代碼
4. 啓動服務,並訪問接口
訪問 /use/print 接口,會發如今日誌中打印出了配置信息
2019-05-24 16:45:04.234  INFO 36687 --- [nio-3801-exec-1] k.s.boot.starter.example.KiteService     : 127.0.0.1:3801複製代碼

順着上面的思路,咱們來看一下官方的 starters 的結構。先來把 Spring Boot 從 github 上 clone 一份下來。用 idea 打開,能夠看到項目結構以下
Spring-boot-starters 中就是官方提供的主要 starters,好比 jdbc、redis、security、web 等等。
咱們拿 spring-boot-starter-data-redis 這個 starter 做爲例子,來講一說官方是怎麼組織項目結構的,以及閱讀源碼的順序應該是怎樣的。
1. 展開 Spring-boot-staters 下的 redis starter,咱們看到目錄結構以下
其中並無 Java 代碼,只有一個 spring.provides 文件,裏面的內容以下:
provides: spring-data-redis,lettuce-core複製代碼
意思就是說,本項目依賴 spring-data-redis 和 lettuce-core 這兩個包,而且在 pom 文件中引用了。其目的就是告知使用者在引用此包的時候,沒必要再引用 provides 中的依賴包了。
2. 而後就是自動註解了,全部 stater 的自動註解類、屬性配置類都放到了 spring-boot-autoconfigure 這個項目下
看到熟悉的 spring.factories 沒有,前面咱們本身實現過。這個內容比較多,咱們只看 redis 相關的
org.springframework.boot.autoconfigure.data.redis.RedisAutoConfiguration,\
org.springframework.boot.autoconfigure.data.redis.RedisReactiveAutoConfiguration,\
org.springframework.boot.autoconfigure.data.redis.RedisRepositoriesAutoConfiguration複製代碼
包含三個自動配置文件,而後順着配置,咱們找到所在 package
而後就能夠開始閱讀代碼了。其餘的 starter 也是一樣的結構。
歡迎工做一到五年的Java工程師朋友們加入Java程序員開發: 721575865
羣內提供免費的Java架構學習資料(裏面有高可用、高併發、高性能及分佈式、Jvm性能調優、Spring源碼,MyBatis,Netty,Redis,Kafka,Mysql,Zookeeper,Tomcat,Docker,Dubbo,Nginx等多個知識點的架構資料)合理利用本身每一分每一秒的時間來學習提高本身,不要再用"沒有時間「來掩飾本身思想上的懶惰!趁年輕,使勁拼,給將來的本身一個交代!
相關文章
相關標籤/搜索