1. 程式人生 > >徒手擼一個 Spring Boot 中的 Starter ,解密自動化配置黑魔法!

徒手擼一個 Spring Boot 中的 Starter ,解密自動化配置黑魔法!

我們使用 Spring Boot,基本上都是沉醉在它 Stater 的方便之中。Starter 為我們帶來了眾多的自動化配置,有了這些自動化配置,我們可以不費吹灰之力就能搭建一個生產級開發環境,有的小夥伴會覺得這個 Starter 好神奇呀!其實 Starter 也都是 Spring + SpringMVC 中的基礎知識點實現的,今天鬆哥就來帶大家自己來擼一個 Starter ,慢慢揭開 Starter 的神祕面紗!

核心知識

其實 Starter 的核心就是條件註解 @Conditional ,當 classpath 下存在某一個 Class 時,某個配置才會生效,前面鬆哥已經帶大家學習過不少 Spring Boot 中的知識點,有的也涉及到原始碼解讀,大夥可能也發現了原始碼解讀時總是會出現條件註解,其實這就是 Starter 配置的核心之一,大夥有興趣可以翻翻歷史記錄,看看鬆哥之前寫的關於 Spring Boot 的文章,這裡我就不再重複介紹了。

定義自己的 Starter

定義

所謂的 Starter ,其實就是一個普通的 Maven 專案,因此我們自定義 Starter ,需要首先建立一個普通的 Maven 專案,建立完成後,新增 Starter 的自動化配置類即可,如下:

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-autoconfigure</artifactId>
    <version>2.1.4.RELEASE</version>
</dependency>

配置完成後,我們首先建立一個 HelloProperties 類,用來接受 application.properties 中注入的值,如下:

@ConfigurationProperties(prefix = "javaboy")
public class HelloProperties {
    private static final String DEFAULT_NAME = "江南一點雨";
    private static final String DEFAULT_MSG = "牧碼小子";
    private String name = DEFAULT_NAME;
    private String msg = DEFAULT_MSG;
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public String getMsg() {
        return msg;
    }
    public void setMsg(String msg) {
        this.msg = msg;
    }
}

這個配置類很好理解,將 application.properties 中配置的屬性值直接注入到這個例項中, @ConfigurationProperties 型別安全的屬性注入,即將 application.properties 檔案中字首為 javaboy 的屬性注入到這個類對應的屬性上, 最後使用時候,application.properties 中的配置檔案,大概如下:

javaboy.name=zhangsan
javaboy.msg=java

關注型別安全的屬性注入,讀者可以參考鬆哥之前的這篇文章:Spring Boot中的yaml配置簡介,這篇文章雖然是講 yaml 配置,但是關於型別安全的屬性注入和 properties 是一樣的。

配置完成 HelloProperties 後,接下來我們來定義一個 HelloService ,然後定義一個簡單的 say 方法, HelloService 的定義如下:

public class HelloService {
    private String msg;
    private String name;
    public String sayHello() {
        return name + " say " + msg + " !";
    }
    public String getMsg() {
        return msg;
    }
    public void setMsg(String msg) {
        this.msg = msg;
    }
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
}

這個很簡單,沒啥好說的。

接下來就是我們的重軸戲,自動配置類的定義,用了很多別人定義的自定義類之後,我們也來自己定義一個自定義類。先來看程式碼吧,一會鬆哥再慢慢解釋:

@Configuration
@EnableConfigurationProperties(HelloProperties.class)
@ConditionalOnClass(HelloService.class)
public class HelloServiceAutoConfiguration {
    @Autowired
    HelloProperties helloProperties;

    @Bean
    HelloService helloService() {
        HelloService helloService = new HelloService();
        helloService.setName(helloProperties.getName());
        helloService.setMsg(helloProperties.getMsg());
        return helloService;
    }
}

關於這一段自動配置,解釋如下:

  • 首先 @Configuration 註解表明這是一個配置類。
  • @EnableConfigurationProperties 註解是使我們之前配置的 @ConfigurationProperties 生效,讓配置的屬性成功的進入 Bean 中。
  • @ConditionalOnClass 表示當專案當前 classpath 下存在 HelloService 時,後面的配置才生效。
  • 自動配置類中首先注入 HelloProperties ,這個例項中含有我們在 application.properties 中配置的相關資料。
  • 提供一個 HelloService 的例項,將 HelloProperties 中的值注入進去。

做完這一步之後,我們的自動化配置類就算是完成了,接下來還需要一個 spring.factories 檔案,那麼這個檔案是幹嘛的呢?大家知道我們的 Spring Boot 專案的啟動類都有一個 @SpringBootApplication 註解,這個註解的定義如下:

@SpringBootConfiguration
@EnableAutoConfiguration
@ComponentScan(excludeFilters = {
        @Filter(type = FilterType.CUSTOM, classes = TypeExcludeFilter.class),
        @Filter(type = FilterType.CUSTOM,
                classes = AutoConfigurationExcludeFilter.class) })
public @interface SpringBootApplication {
}

大家看到這是一個組合註解,其中的一個組合項就是 @EnableAutoConfiguration ,這個註解是幹嘛的呢?

@EnableAutoConfiguration 表示啟用 Spring 應用程式上下文的自動配置,該註解會自動匯入一個名為 AutoConfigurationImportSelector 的類,而這個類會去讀取一個名為 spring.factories 的檔案, spring.factories 中則定義需要載入的自動化配置類,我們開啟任意一個框架的 Starter ,都能看到它有一個 spring.factories 檔案,例如 MyBatis 的 Starter 如下:

那麼我們自定義 Starter 當然也需要這樣一個檔案,我們首先在 Maven 專案的 resources 目錄下建立一個名為 META-INF 的資料夾,然後在資料夾中建立一個名為 spring.factories 的檔案,檔案內容如下:

org.springframework.boot.autoconfigure.EnableAutoConfiguration=org.javaboy.mystarter.HelloServiceAutoConfiguration

在這裡指定我們的自動化配置類的路徑即可。

如此之後我們的自動化配置類就算完成了。

本地安裝

如果在公司裡,大夥可能需要將剛剛寫好的自動化配置類打包,然後上傳到 Maven 私服上,供其他同事下載使用,我這裡就簡單一些,我就不上傳私服了,我將這個自動化配置類安裝到本地倉庫,然後在其他專案中使用即可。安裝方式很簡單,在 IntelliJ IDEA 中,點選右邊的 Maven Project ,然後選擇 Lifecycle 中的 install ,雙擊即可,如下:

雙擊完成後,這個 Starter 就安裝到我們本地倉庫了,當然小夥伴也可以使用 Maven 命令去安裝。

使用 Starter

接下來,我們來新建一個普通的 Spring Boot 工程,這個 Spring Boot 建立成功之後,加入我們自定義 Starter 的依賴,如下:

<dependency>
    <groupId>org.javaboy</groupId>
    <artifactId>mystarter</artifactId>
    <version>1.0-SNAPSHOT</version>
</dependency>

此時我們引入了上面自定義的 Starter ,也即我們專案中現在有一個預設的 HelloService 例項可以使用,而且關於這個例項的資料,我們還可以在 application.properties 中進行配置,如下:

javaboy.name=牧碼小子
javaboy.msg=java

配置完成後,方便起見,我這裡直接在單元測試方法中注入 HelloSerivce 例項來使用,程式碼如下:

@RunWith(SpringRunner.class)
@SpringBootTest
public class UsemystarterApplicationTests {

    @Autowired
    HelloService helloService;
    @Test
    public void contextLoads() {
        System.out.println(helloService.sayHello());
    }
}

執行單元測試方法,列印日誌如下:

好了,一個簡單的自動化配置類我們就算完成了,是不是很簡單!

總結

本文主要帶領小夥伴自己徒手擼一個 Starter ,使用這種方式幫助大家揭開 Starter 的神祕面紗!大夥有問題可以留言討論。

本文的案例,鬆哥已經上傳到 GitHub上了,地址:https://github.com/lenve/javaboy-code-samples 。

關注公眾號牧碼小子,專注於 Spring Boot+微服務,定期視訊教程分享,關注後回覆 Java ,領取鬆哥為你精心準備的 Java 乾貨!

相關推薦

徒手一個 Spring Boot Starter 解密自動化配置魔法

我們使用 Spring Boot,基本上都是沉醉在它 Stater 的方便之中。Starter 為我們帶來了眾多的自動化配置,有了這些自動化配置,我們可以不費吹灰之力就能搭建一個生產級開發環境,有的小夥伴會覺得這個 Starter 好神奇呀!其實 Starter 也都是 Spring + SpringMVC

Spring BootStarter是什麽

bsp ram apache .so plugin tab 我們 redis gin 比如我們要在Spring Boot中引入Web MVC的支持時,我們通常會引入這個模塊spring-boot-starter-web,而這個模塊如果解壓包出來會發現裏面什麽都沒有,只定義了

Spring-boot@ConfigurationProperties@Value@PropertySource

1.利用@ConfigurationProperties獲取配置的值,@ConfigurationProperties是springboot提供的基於安全型別的配置放置。     application.properties spring.redis.host=127.0.0.1

創建一個 Spring Boot 項目你會幾種方法?

text 通過 們的 但是 stc 自動化 新項目 war group 我最早是 2016 年底開始寫 Spring Boot 相關的博客,當時使用的版本還是 1.4.x ,文章發表在 CSDN 上,閱讀量最大的一篇有 42W+,如下圖: 2017 年由於種種原因,就沒有

曹工談Spring BootSpring boot怎麼進行外部化配置一不留神摔一跤;一路debug原來是我太年輕了

# spring boot中怎麼進行外部化配置,一不留神摔一跤;一路debug,原來是我太年輕了 # 背景 我們公司這邊,目前都是spring boot專案,沒有引入spring cloud config,也就是說,配置檔案,還是放在resources下面的,為了區分多環境,是採用了profile這種方式

spring-boot application.properties的各種配置

ecif creat ati .sh drive fig nag ide cte ###########################################################datasource connect mysql#############

Spring boot 通過profile屬性指定配置檔案

1、多Profile檔案 1 通過在配置檔案中配置 一個專案在開發的過程中可能存在搓個配置檔案一個是生產環境,一個是測試環境,那麼如何去指定呢? 我們知道在主配置檔案編寫的時候,檔名可以是 application-{profile}.properties/yml。 預設使用a

Spring Boot的快取支援——註解配置與EhCache使用

一 點睛 隨著時間的積累,應用的使用使用者不斷增加,資料規模也越來越大,往往資料庫查詢操作會成為影響使用者使用體驗的瓶頸,此時使用快取往往是解決這一問題非常好的手段之一。 Spring 3開始提供了強大的基於註解的快取支援,可以通過註解配置方式低侵入的給原有Spring應

spring boot使用druid多資料來源配置Mybatis

首選必須吐槽下,不知道是我讀書太少,還是Spring boot確實對druid和mybatis的支援不好,奈何用怪了druid的資料庫、sql、url監控和mybatis資料來源,只有硬著頭皮上了。 相關配置 常例先給出build.gradle配置 "or

spring boot的log4j2程式設計式配置(Programmatic Configuration)注意事項

1、在入口類中,不可有如下的類變數申明: private static final Logger logger = LogManager.getLogger(App.class); 2、如果採取程式設

Spring-Boot-操作-Redis三種方案全解析

在 Redis 出現之前,我們的快取框架各種各樣,有了 Redis ,快取方案基本上都統一了,關於 Redis,鬆哥之前有一個系列教程,尚不瞭解 Redis 的小夥伴可以參考這個教程: Redis 教程合集 使用 Java 操作 Redis 的方案很多,Jedis 是目前較為流行的一種方案,除了 Jedi

Spring Boot 整合 Shiro 兩種方式全總結

在 Spring Boot 中做許可權管理,一般來說,主流的方案是 Spring Security ,但是,僅僅從技術角度來說,也可以使用 Shiro。 今天鬆哥就來和大家聊聊 Spring Boot 整合 Shiro 的話題! 一般來說,Spring Security 和 Shiro 的比較如下: Sp

Spring Boot 配置定時任務實現多線程操作

pre log pri http code china 部分 多線程操作 .net 參考的代碼部分 https://git.oschina.net/jokerForTao/spring_boot_schedule 一目了然!Spring Boot 中配置定時任務,實現

使用Logstash同步數據至ElasticsearchSpring Boot集成Elasticsearch實現搜索

開啟 stash auto -a zab rest driver tid list 安裝logstash、同步數據至ElasticSearch 為什麽使用logstash來同步,CSDN上有一篇文章簡要的分析了以下幾種同步工具的優缺點:https://blog.csdn.

spring bootsecurity安全退出如何跳轉指定頁面iframe與安全器相容性問題

分享一下這這次專案中自己學到的一些東西(還沒學完,技術很菜,寫的有問題希望大家指出來,希望大家可以一起學習,一起努力)       在WebSecurityConfig中配置:        http.log

Spring Boot 單檔案多檔案上傳以及將檔案寫入響應

單檔案上傳 配置檔案設定 @Component public class MultipartConfig { @Bean public MultipartConfigElement multipartConfigElement(){ Multipa

spring boot返回date為EEE MMM dd HH:mm:ss z yyyy(Fri Oct 26 00:00:00 CST 2018)轉換為 yyyy-MM-dd HH:mm:ss

問題:資料庫中日期型別為timestamp,實體類中定義為date,返回給到前端時變為了EEE MMM dd HH:mm:ss z yyyy(Fri Oct 26 00:00:00 CST 2018)格式,而我們需要的是yyyy-MM-dd HH:mm:ss。 解決方式: 在實體類定義中,定

Spring Boot使用thymeleaf以及各種取值判斷選擇擷取等方式

Spring Boot中使用thymeleaf Spring Boot支援FreeMarker、Groovy、Thymeleaf和Mustache四種模板解析引擎,官方推薦使用Thymeleaf。 spring-boot-starter-thymeleaf 在Spring B

在IDEA建立第一個Spring Boot應用

1.建立springboot專案 Next後邊的步驟都是平時用的,這裡就不再放圖出來了 2.編寫一個簡單程式碼進行測試 在SpringbootApplication.java中: package cn.ysjh.springboot2; import org.springfr

spring bootspring security實現單點登入傳統模式(一)

單點登入是什麼? 一個系統中可能會引用別的很多系統。單點登入就是解決,一次登入,就可以訪問所有的系統。 每次瀏覽器向一個域名傳送http請求,會去查詢域名的cookie資訊拼接到http的header中傳送到伺服器。 cookie不能跨域。這個域是瀏覽器請求的域名,哪怕他們都是訪問一