SpringBoot自動配置探究

  1. @SpringBootApplication

    • @SpringBootApplication表示SpringBoot應用,標註在某個類上說明這個類是SpringBoot的主配置類,SpringBoot就運行這個類的main方法來啓動應用;
    • @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:表示是一個SpringBoot的配置類
        @Target({ElementType.TYPE})
        @Retention(RetentionPolicy.RUNTIME)
        @Documented
        @Configuration
        public @interface SpringBootConfiguration {
        }
        • @Configuration:Spring定義的註解,標註在類上表示這是個配置類,使用註解定義配置類來取代配置文件;配置類也是容器中的一個組件
          @Target({ElementType.TYPE})
          @Retention(RetentionPolicy.RUNTIME)
          @Documented
          @Component
          public @interface Configuration {
              @AliasFor(
                  annotation = Component.class
              )
              String value() default "";
          }
      • @EnableAutoConfiguration:開啓自動配置功能
        @AutoConfigurationPackage
        @Import(AutoConfigurationImportSelector.class)
        public @interface EnableAutoConfiguration {
              .......
        }
        • @AutoConfigurationPackage:自動配置包;Spring的底層註解@Import,表示給容器導入一個組件,
          @Import(AutoConfigurationPackages.Registrar.class)
          public @interface AutoConfigurationPackage {}
          • Registrar.class:new PackageImports(metadata).getPackageNames().toArray(new String[0])得到註解類@SpringBootApplication所在的包名,把包下面的全部類都加入到Spring容器中
                    @Override
                    public void registerBeanDefinitions(AnnotationMetadata metadata, BeanDefinitionRegistry registry) {
                        register(registry, new PackageImports(metadata).getPackageNames().toArray(new String[0]));
                    }
        • AutoConfigurationImportSelector.class:給容器導入組件的自動配置類,並配置好這些組件,有了自動配置類,就免去了手動編寫配置注入功能組件等操做;
              @Override
              public String[] selectImports(AnnotationMetadata annotationMetadata) {
                  if (!isEnabled(annotationMetadata)) {
                      return NO_IMPORTS;
                  }
                  AutoConfigurationEntry autoConfigurationEntry = getAutoConfigurationEntry(annotationMetadata);
                  return StringUtils.toStringArray(autoConfigurationEntry.getConfigurations());
              }
              protected AutoConfigurationEntry getAutoConfigurationEntry(AnnotationMetadata annotationMetadata) {
                  if (!isEnabled(annotationMetadata)) {
                      return EMPTY_ENTRY;
                  }
                  AnnotationAttributes attributes = getAttributes(annotationMetadata);
                  //得到全部組件的自動配置類,包括mq的RabbitAutoConfiguration、redis的RedisAutoConfiguration等等
                  List<String> configurations = getCandidateConfigurations(annotationMetadata, attributes);
                  configurations = removeDuplicates(configurations);
                  Set<String> exclusions = getExclusions(annotationMetadata, attributes);
                  checkExcludedClasses(configurations, exclusions);
                  configurations.removeAll(exclusions);
                  configurations = getConfigurationClassFilter().filter(configurations);
                  fireAutoConfigurationImportEvents(configurations, exclusions);
                  return new AutoConfigurationEntry(configurations, exclusions);
              }
              protected List<String> getCandidateConfigurations(AnnotationMetadata metadata, AnnotationAttributes attributes) {
                  List<String> configurations = SpringFactoriesLoader.loadFactoryNames(getSpringFactoriesLoaderFactoryClass(),
                          getBeanClassLoader());
                  Assert.notEmpty(configurations, "No auto configuration classes found in META-INF/spring.factories. If you "
                          + "are using a custom packaging, make sure that file is correct.");
                  return configurations;
              }
              public static final String FACTORIES_RESOURCE_LOCATION = "META-INF/spring.factories";
              //loadFactoryNames調用了loadSpringFactories(),它加載了spring.factories配置的信息,根據Map鍵值對獲取信息,而後導入到容器中,在spring-boot-autoconfigure-2.3.0.RELEASE.jar中定義了文件
              public static List<String> loadFactoryNames(Class<?> factoryType, @Nullable ClassLoader classLoader) {
                  String factoryTypeName = factoryType.getName();
                  return loadSpringFactories(classLoader).getOrDefault(factoryTypeName, Collections.emptyList());
              }
          • 好比MVC的自動配置,就使用了WebMvcAutoConfiguration類,下面的代碼就是給容器添加了視圖解析器
                    @Bean
                    @ConditionalOnBean(View.class)
                    @ConditionalOnMissingBean
                    public BeanNameViewResolver beanNameViewResolver() {
                        BeanNameViewResolver resolver = new BeanNameViewResolver();
                        resolver.setOrder(Ordered.LOWEST_PRECEDENCE - 10);
                        return resolver;
                    }
  2. @ConfigurationProperties

    • 批量注入配置文件中的屬性值,默認從全局配置文件中獲取:在配置文件yml或properties定義值,使用該註解給配置類賦值
      //application.properties
      //可以使用$佔位符定義值,好比person.name=張三${random.int},person.maps.k2=${person.name}_v2;${person.name:hello}表示person_name沒有定義,則使用默認值hello
      //結果name爲張三12afa124,k2爲張三12afa124_v2
      person.name=張三
      person.age=11
      person.maps.k1=v1
      person.maps.k2=v2
      person.list=a,b
      //定義配置類並加入容器中,將上面的值按照鍵值對一一賦值
      @Component
      @ConfigurationProperties(prefix = "person")
      public class Person {
          private String name;
          private Integer age;
          private Map<String,Object> maps;
          private List<String> list;
      }
    • @ConfigurationProperties和@Value對比
      • @ConfigurationProperties批量注入文件的屬性值;@Value一個個指定;
      • @ConfigurationProperties支持鬆散綁定(person.lastName;person.last_name;person.last-name;PERSON_LAST_NAME);@Value不支持;
      • @ConfigurationProperties不支持SpEL;@Value支持,好比@Value("#{10}");
      • @ConfigurationProperties支持JSR303數據校驗,好比在配置類上添加@Validated,而後在屬性上添加@Email,它表示屬性值必須符合Email格式;@Value不支持,同時也不支持複雜類型封裝,好比Map;
      • 若是須要JavaBean和配置文件值進行映射,使用@ConfigurationProperties;若是是單個值使用@Value;
    • @PropertySource:表示加載指定路徑的配置文件
      @PropertySource(value = {"classpath:person.properties"})
    • @ImportResource:導入Spring的配置文件,讓配置文件裏面的內容生效,好比咱們寫了bean.xml,裏面使用<bean>建立了一個實例,但沒有加入容器中
      @ImportResource(locations = {"classpath:bean.xml"})
      • 使用@Configuration和@Bean來給容器添加組件,代替上面的xml文件中方式
        @Configuration
        public class MyConfiguration {
            @Bean
            public HelloService helloService(){
                return new HelloService();
            }
        }
  3. 自動配置原理

    1. SpringBoot啓動加載主配置類(@SpringBootApplication),開啓了自動配置功能(@EnableAutoConfiguration);
    2. 自動配置使用了@Import(AutoConfigurationImportSelector.class)導入了組件,可查看getAutoConfigurationEntry()的getCandidateConfigurations(AnnotationMetadata metadata, AnnotationAttributes attributes)返回的集合,即候選的配置;
    3. SpringFactoriesLoader.loadFactoryNames()掃描全部jar包路徑下的META-INF/spring.factories,把掃描到的文件的內容包裝成properties對象,從properties中獲取到EnableAutoConfiguration.class類對應的值,而後把它們添加到容器中;
      #spring-boot-autoconfigure-2.3.0.RELEASE.jar中spring.factories的信息,EnableAutoConfiguration後面每一個xxxAutoConfiguration都是一個組件,都加入到容器中
      # Auto Configure
      org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
      org.springframework.boot.autoconfigure.admin.SpringApplicationAdminJmxAutoConfiguration,\
      org.springframework.boot.autoconfigure.aop.AopAutoConfiguration,\
      org.springframework.boot.autoconfigure.amqp.RabbitAutoConfiguration,\
      org.springframework.boot.autoconfigure.batch.BatchAutoConfiguration,\
      org.springframework.boot.autoconfigure.cache.CacheAutoConfiguration,\
      org.springframework.boot.autoconfigure.cassandra.CassandraAutoConfiguration,\
      org.springframework.boot.autoconfigure.context.ConfigurationPropertiesAutoConfiguration,\
      org.springframework.boot.autoconfigure.context.LifecycleAutoConfiguration,\
      org.springframework.boot.autoconfigure.context.MessageSourceAutoConfiguration,\
      org.springframework.boot.autoconfigure.context.PropertyPlaceholderAutoConfiguration,\
      org.springframework.boot.autoconfigure.couchbase.CouchbaseAutoConfiguration,\
      org.springframework.boot.autoconfigure.dao.PersistenceExceptionTranslationAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.cassandra.CassandraDataAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.cassandra.CassandraReactiveDataAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.cassandra.CassandraReactiveRepositoriesAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.cassandra.CassandraRepositoriesAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.couchbase.CouchbaseDataAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.couchbase.CouchbaseReactiveDataAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.couchbase.CouchbaseReactiveRepositoriesAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.couchbase.CouchbaseRepositoriesAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.elasticsearch.ElasticsearchDataAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.elasticsearch.ElasticsearchRepositoriesAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.elasticsearch.ReactiveElasticsearchRepositoriesAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.elasticsearch.ReactiveElasticsearchRestClientAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.jdbc.JdbcRepositoriesAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.jpa.JpaRepositoriesAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.ldap.LdapRepositoriesAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.mongo.MongoDataAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.mongo.MongoReactiveDataAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.mongo.MongoReactiveRepositoriesAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.mongo.MongoRepositoriesAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.neo4j.Neo4jDataAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.neo4j.Neo4jRepositoriesAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.solr.SolrRepositoriesAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.r2dbc.R2dbcDataAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.r2dbc.R2dbcRepositoriesAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.r2dbc.R2dbcTransactionManagerAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.redis.RedisAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.redis.RedisReactiveAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.redis.RedisRepositoriesAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.rest.RepositoryRestMvcAutoConfiguration,\
      org.springframework.boot.autoconfigure.data.web.SpringDataWebAutoConfiguration,\
      org.springframework.boot.autoconfigure.elasticsearch.ElasticsearchRestClientAutoConfiguration,\
      org.springframework.boot.autoconfigure.flyway.FlywayAutoConfiguration,\
      org.springframework.boot.autoconfigure.freemarker.FreeMarkerAutoConfiguration,\
      org.springframework.boot.autoconfigure.groovy.template.GroovyTemplateAutoConfiguration,\
      org.springframework.boot.autoconfigure.gson.GsonAutoConfiguration,\
      org.springframework.boot.autoconfigure.h2.H2ConsoleAutoConfiguration,\
      org.springframework.boot.autoconfigure.hateoas.HypermediaAutoConfiguration,\
      org.springframework.boot.autoconfigure.hazelcast.HazelcastAutoConfiguration,\
      org.springframework.boot.autoconfigure.hazelcast.HazelcastJpaDependencyAutoConfiguration,\
      org.springframework.boot.autoconfigure.http.HttpMessageConvertersAutoConfiguration,\
      org.springframework.boot.autoconfigure.http.codec.CodecsAutoConfiguration,\
      org.springframework.boot.autoconfigure.influx.InfluxDbAutoConfiguration,\
      org.springframework.boot.autoconfigure.info.ProjectInfoAutoConfiguration,\
      org.springframework.boot.autoconfigure.integration.IntegrationAutoConfiguration,\
      org.springframework.boot.autoconfigure.jackson.JacksonAutoConfiguration,\
      org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration,\
      org.springframework.boot.autoconfigure.jdbc.JdbcTemplateAutoConfiguration,\
      org.springframework.boot.autoconfigure.jdbc.JndiDataSourceAutoConfiguration,\
      org.springframework.boot.autoconfigure.jdbc.XADataSourceAutoConfiguration,\
      org.springframework.boot.autoconfigure.jdbc.DataSourceTransactionManagerAutoConfiguration,\
      org.springframework.boot.autoconfigure.jms.JmsAutoConfiguration,\
      org.springframework.boot.autoconfigure.jmx.JmxAutoConfiguration,\
      org.springframework.boot.autoconfigure.jms.JndiConnectionFactoryAutoConfiguration,\
      org.springframework.boot.autoconfigure.jms.activemq.ActiveMQAutoConfiguration,\
      org.springframework.boot.autoconfigure.jms.artemis.ArtemisAutoConfiguration,\
      org.springframework.boot.autoconfigure.jersey.JerseyAutoConfiguration,\
      org.springframework.boot.autoconfigure.jooq.JooqAutoConfiguration,\
      org.springframework.boot.autoconfigure.jsonb.JsonbAutoConfiguration,\
      org.springframework.boot.autoconfigure.kafka.KafkaAutoConfiguration,\
      org.springframework.boot.autoconfigure.availability.ApplicationAvailabilityAutoConfiguration,\
      org.springframework.boot.autoconfigure.ldap.embedded.EmbeddedLdapAutoConfiguration,\
      org.springframework.boot.autoconfigure.ldap.LdapAutoConfiguration,\
      org.springframework.boot.autoconfigure.liquibase.LiquibaseAutoConfiguration,\
      org.springframework.boot.autoconfigure.mail.MailSenderAutoConfiguration,\
      org.springframework.boot.autoconfigure.mail.MailSenderValidatorAutoConfiguration,\
      org.springframework.boot.autoconfigure.mongo.embedded.EmbeddedMongoAutoConfiguration,\
      org.springframework.boot.autoconfigure.mongo.MongoAutoConfiguration,\
      org.springframework.boot.autoconfigure.mongo.MongoReactiveAutoConfiguration,\
      org.springframework.boot.autoconfigure.mustache.MustacheAutoConfiguration,\
      org.springframework.boot.autoconfigure.orm.jpa.HibernateJpaAutoConfiguration,\
      org.springframework.boot.autoconfigure.quartz.QuartzAutoConfiguration,\
      org.springframework.boot.autoconfigure.r2dbc.R2dbcAutoConfiguration,\
      org.springframework.boot.autoconfigure.rsocket.RSocketMessagingAutoConfiguration,\
      org.springframework.boot.autoconfigure.rsocket.RSocketRequesterAutoConfiguration,\
      org.springframework.boot.autoconfigure.rsocket.RSocketServerAutoConfiguration,\
      org.springframework.boot.autoconfigure.rsocket.RSocketStrategiesAutoConfiguration,\
      org.springframework.boot.autoconfigure.security.servlet.SecurityAutoConfiguration,\
      org.springframework.boot.autoconfigure.security.servlet.UserDetailsServiceAutoConfiguration,\
      org.springframework.boot.autoconfigure.security.servlet.SecurityFilterAutoConfiguration,\
      org.springframework.boot.autoconfigure.security.reactive.ReactiveSecurityAutoConfiguration,\
      org.springframework.boot.autoconfigure.security.reactive.ReactiveUserDetailsServiceAutoConfiguration,\
      org.springframework.boot.autoconfigure.security.rsocket.RSocketSecurityAutoConfiguration,\
      org.springframework.boot.autoconfigure.security.saml2.Saml2RelyingPartyAutoConfiguration,\
      org.springframework.boot.autoconfigure.sendgrid.SendGridAutoConfiguration,\
      org.springframework.boot.autoconfigure.session.SessionAutoConfiguration,\
      org.springframework.boot.autoconfigure.security.oauth2.client.servlet.OAuth2ClientAutoConfiguration,\
      org.springframework.boot.autoconfigure.security.oauth2.client.reactive.ReactiveOAuth2ClientAutoConfiguration,\
      org.springframework.boot.autoconfigure.security.oauth2.resource.servlet.OAuth2ResourceServerAutoConfiguration,\
      org.springframework.boot.autoconfigure.security.oauth2.resource.reactive.ReactiveOAuth2ResourceServerAutoConfiguration,\
      org.springframework.boot.autoconfigure.solr.SolrAutoConfiguration,\
      org.springframework.boot.autoconfigure.task.TaskExecutionAutoConfiguration,\
      org.springframework.boot.autoconfigure.task.TaskSchedulingAutoConfiguration,\
      org.springframework.boot.autoconfigure.thymeleaf.ThymeleafAutoConfiguration,\
      org.springframework.boot.autoconfigure.transaction.TransactionAutoConfiguration,\
      org.springframework.boot.autoconfigure.transaction.jta.JtaAutoConfiguration,\
      org.springframework.boot.autoconfigure.validation.ValidationAutoConfiguration,\
      org.springframework.boot.autoconfigure.web.client.RestTemplateAutoConfiguration,\
      org.springframework.boot.autoconfigure.web.embedded.EmbeddedWebServerFactoryCustomizerAutoConfiguration,\
      org.springframework.boot.autoconfigure.web.reactive.HttpHandlerAutoConfiguration,\
      org.springframework.boot.autoconfigure.web.reactive.ReactiveWebServerFactoryAutoConfiguration,\
      org.springframework.boot.autoconfigure.web.reactive.WebFluxAutoConfiguration,\
      org.springframework.boot.autoconfigure.web.reactive.error.ErrorWebFluxAutoConfiguration,\
      org.springframework.boot.autoconfigure.web.reactive.function.client.ClientHttpConnectorAutoConfiguration,\
      org.springframework.boot.autoconfigure.web.reactive.function.client.WebClientAutoConfiguration,\
      org.springframework.boot.autoconfigure.web.servlet.DispatcherServletAutoConfiguration,\
      org.springframework.boot.autoconfigure.web.servlet.ServletWebServerFactoryAutoConfiguration,\
      org.springframework.boot.autoconfigure.web.servlet.error.ErrorMvcAutoConfiguration,\
      org.springframework.boot.autoconfigure.web.servlet.HttpEncodingAutoConfiguration,\
      org.springframework.boot.autoconfigure.web.servlet.MultipartAutoConfiguration,\
      org.springframework.boot.autoconfigure.web.servlet.WebMvcAutoConfiguration,\
      org.springframework.boot.autoconfigure.websocket.reactive.WebSocketReactiveAutoConfiguration,\
      org.springframework.boot.autoconfigure.websocket.servlet.WebSocketServletAutoConfiguration,\
      org.springframework.boot.autoconfigure.websocket.servlet.WebSocketMessagingAutoConfiguration,\
      org.springframework.boot.autoconfigure.webservices.WebServicesAutoConfiguration,\
      org.springframework.boot.autoconfigure.webservices.client.WebServiceTemplateAutoConfiguration
    4. 每個自動配置類進行自動配置,以HttpEncodingAutoConfiguration爲例:
      @Configuration(proxyBeanMethods = false)//表示這是一個配置類
      @EnableConfigurationProperties(ServerProperties.class)//啓動指定類的ConfigurationProperties功能,將配置文件中對應的值和ServerProperties綁定起來,而後加入到IOC容器中
      @ConditionalOnWebApplication(type = ConditionalOnWebApplication.Type.SERVLET)//Spring底層@Conditional註解,根據不一樣的條件,若是知足指定的條件,整個配置類裏面的配置就會生效;判斷當前應用是不是web應用,若是是,當前配置類生效
      @ConditionalOnClass(CharacterEncodingFilter.class)//判斷當前項目有沒有這個CharacterEncodingFilter類;SpringMVC中進行亂碼解決的過濾器
      @ConditionalOnProperty(prefix = "server.servlet.encoding", value = "enabled", matchIfMissing = true)//判斷配置文件中是否存在某個配置,server.servlet.encoding,若是不存在,判斷也是成立的
      public class HttpEncodingAutoConfiguration {
      
          private final Encoding properties;//已經和SpringBoot的配置文件映射了,即ServerProperties中的Encoding對象
          //只有一個有參構造器的狀況下,參數的值從容器中拿,即@EnableConfigurationProperties(ServerProperties.class)
          public HttpEncodingAutoConfiguration(ServerProperties properties) {
              this.properties = properties.getServlet().getEncoding();//從ServerProperties中獲取Encoding對象並進行實例化
          }
          @Bean //給容器中添加一個組件CharacterEncodingFilter
          @ConditionalOnMissingBean
          public CharacterEncodingFilter characterEncodingFilter() {
              CharacterEncodingFilter filter = new OrderedCharacterEncodingFilter();
              filter.setEncoding(this.properties.getCharset().name());//從properties中獲取值並賦值
              filter.setForceRequestEncoding(this.properties.shouldForce(Encoding.Type.REQUEST));
              filter.setForceResponseEncoding(this.properties.shouldForce(Encoding.Type.RESPONSE));
              return filter;
          }
      }
      //從配置文件中獲取指定的值和bean的屬性進行綁定(全部在配置文件中能配置的屬性都是在xxxProperties類中封裝着;配置文件能配置什麼參數就參照該功能對應的屬性類)
      @ConfigurationProperties(prefix = "server", ignoreUnknownFields = true)
      public class ServerProperties {}
    5. 一旦這個自動配置類生效,就會給容器添加各類組件@Bean,這些組件的屬性是從對應的properties類中獲取的,每一個properties類裏面的每個屬性又是和配置文件綁定的;
  4. @Conditional派生註解

    • 做用在某個類上,表示必須是@Condition指定的條件成立纔會給容器中添加組件,配置類裏面的內容才生效;
    • 經過配置文件裏面的debug=true在控制檯打印出自動配置的啓動和沒有啓用的報告進行查看:
      ============================
      CONDITIONS EVALUATION REPORT條件評估報告
      ============================
      
      
      Positive matches:(自動配置類啓用的)
      -----------------
      
         AopAutoConfiguration matched:
            - @ConditionalOnProperty (spring.aop.auto=true) matched (OnPropertyCondition)
         ...........
      Negative matches:(自動配置類沒有啓用的)
      -----------------
      
         ActiveMQAutoConfiguration:
            Did not match:
               - @ConditionalOnClass did not find required class 'javax.jms.ConnectionFactory' (OnClassCondition)
         ...........
相關文章
相關標籤/搜索