Springboot自動配置與@Configuration配置類詳解

@Configuration

註意點1

配置類(@Configuration下的這個類)其實相當於一個工廠, 標註 @Bean 註解的方法相當於工廠方法

考慮有如下例子:

    @Configuration
    // 註意點1: 配置類其實相當於一個工廠, 標註 @Bean 註解的方法相當於工廠方法
    static class MyConfig {
        @Bean
        public Bean1 bean1() {
            System.out.println("bean1()");
            return new Bean1();
        }

將來如果要生成一個bean1的實例,首先要找到一個叫myConfig的bean(是一個工廠),再調用這個工廠裡面叫"bean1"的方法,生成bean1對象。

對於沒有static修飾的@Bean工廠方法,叫做實例工廠,需要先把配置類myConfig實例創建出來;有static修飾的是靜態方法,隻需要拿到myConfig的類對象,然後直接.方法就行。

註意點2

@Bean 不支持方法重載, 如果有多個重載方法, 僅有一個能入選為工廠方法

    @Configuration
    @MapperScan("aaa")
    // 註意點1: 配置類其實相當於一個工廠, 標註 @Bean 註解的方法相當於工廠方法
    static class MyConfig {
        // 註意點2: @Bean 不支持方法重載, 如果有多個重載方法, 僅有一個能入選為工廠方法
        @Bean
        public Bean1 bean1() {
            System.out.println("bean1()");
            return new Bean1();
        }
        @Bean
        public Bean1 bean1(@Value("${java.class.version}") String a) {
            System.out.println("bean1(" + a + ")");
            return new Bean1();
        }
        @Bean
        public Bean1 bean1(@Value("${java.class.version}") String a, @Value("${JAVA_HOME}") String b) {
            System.out.println("bean1(" + a + ", " + b + ")");
            return new Bean1();
        }

如果有重載方法,參數中需要註入的值越多,優先級越高。

註意點3

@Configuration不能隨便刪除,因為默認會為標註的類生成代理, 其目的是保證 @Bean 方法相互調用時, 仍然能保證其單例特性

註意點4

@Configuration 中如果含有 BeanFactory 後處理器, 則實例工廠方法會導致 MyConfig 提前創建, 造成其依賴註入失敗,解決方法是該用靜態工廠方法或直接為 @Bean 的方法參數依賴註入, 針對 MapperScanner 可以改用註解方式

springboot自動配置

@SpringBootApplication 是一個組合註解,由 @ComponentScan、@EnableAutoConfiguration 和 @SpringBootConfiguration 組成:

@SpringBootConfiguration 與普通 @Configuration 相比,唯一區別是前者要求整個 app 中隻出現一次,因為要根據它斷定主配置類,根據主配置類才能找到整個程序的入口

@ComponentScan:組件掃描

excludeFilters – 用來在組件掃描時進行排除,也會排除自動配置類

@EnableAutoConfiguration 也是一個組合註解,由下面註解組成

@AutoConfigurationPackage – 用來記住掃描的起始包,也就是記錄被它標註的類所在的包

@Import(AutoConfigurationImportSelector.class) 用來加載 META-INF/spring.factories 中的自動配置類(自動配置類就是如果主配置類沒有配置的就會去用自動配置類)

為什麼不使用 @Import 直接引入自動配置類

有兩個原因:

  • 讓主配置類和自動配置類變成瞭強耦合,主配置類不應該知道有哪些從屬配置
  • 直接用 @Import(自動配置類.class),引入的配置解析優先級較高,自動配置類的解析應該在主配置沒提供時作為默認配置

因此,采用瞭 @Import(AutoConfigurationImportSelector.class)

  • AutoConfigurationImportSelector.class 去讀取 META-INF/spring.factories 中的自動配置類,實現瞭弱耦合。
  • 另外 AutoConfigurationImportSelector.class 實現瞭 DeferredImportSelector 接口,讓自動配置的解析晚於主配置的解析

到此這篇關於Springboot自動配置與@Configuration配置類詳解的文章就介紹到這瞭,更多相關Springboot自動配置內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: