SpringBoot Starter

圖片

前言

我們都知道锉屈,Spring的功能非常強(qiáng)大,但也有些弊端耻涛。比如:我們需要手動(dòng)去配置大量的參數(shù),沒有默認(rèn)值瘟檩,需要我們管理大量的jar包和它們的依賴抹缕。

為了提升Spring項(xiàng)目的開發(fā)效率,簡(jiǎn)化一些配置墨辛,Spring官方引入了SpringBoot卓研。

當(dāng)然,引入SpringBoot還有其他原因睹簇,在這里就不過多描述了奏赘。

本文重點(diǎn)跟大家一起聊聊SpringBootstarter機(jī)制,因?yàn)樗匾恕?/p>

圖片

1 為什么要用starter带膀?

SpringBoot還沒有出來之前志珍,我們使用Spring開發(fā)項(xiàng)目。如果程序需要連接數(shù)據(jù)庫垛叨,我們一般會(huì)使用HibernateMybatisORM框架伦糯,這里我以Mybatis為例,具體的操作步驟如下:

  1. 到maven倉庫去找需要引入的mybatis jar包嗽元,選取合適的版本敛纲。
  2. 到maven倉庫去找mybatis-spring整合的jar包,選取合適的版本剂癌。
  3. 在spring的applicationContext.xml文件中配置dataSource和mybatis相關(guān)信息淤翔。

當(dāng)然有些朋友可能會(huì)指正,不是還需要引入數(shù)據(jù)庫驅(qū)動(dòng)包嗎佩谷?

確實(shí)需要引入旁壮,但數(shù)據(jù)庫驅(qū)動(dòng)有很多监嗜,比如:mysql、oracle抡谐、sqlserver裁奇,這不屬于mybatis的范疇,使用者可以根據(jù)項(xiàng)目的實(shí)際情況單獨(dú)引入麦撵。

如果程序只是需要連接數(shù)據(jù)庫這一個(gè)功能還好刽肠,按上面的步驟做基本可以滿足需求。但是免胃,連接數(shù)據(jù)庫可能只是龐大的項(xiàng)目體系中一個(gè)環(huán)節(jié)音五,實(shí)際項(xiàng)目中往往更復(fù)雜,需要引入更多的功能羔沙,比如:連接redis躺涝、連接mongodb、使用rocketmq扼雏、使用excel功能等等诞挨。

引入這些功能的話,需要再把上面的步驟再重復(fù)一次呢蛤,工作量無形當(dāng)中增加了不少惶傻,而且有很多重復(fù)的工作

另外其障,還是有個(gè)問題银室,每次到要到maven中找合適的版本,如果哪次找的mybatis.jar包 和 mybatis-spring.jar包版本不兼容励翼,程序不是會(huì)出現(xiàn)問題蜈敢?

SpringBoot為了解決以上兩個(gè)問題引入了starter機(jī)制

2 starter有哪些要素汽抚?

我們首先一起看看mybatis-spring-boot-starter.jar是如何定義的抓狭。

圖片

可以看到它的META-INF目錄下只包含了:

  • pom.protperties 配置maven所需的項(xiàng)目version、groupId和artifactId造烁。
  • pom.xml 配置所依賴的jar包否过。
  • MANIFEST.MF 這個(gè)文件描述了該Jar文件的很多信息。
  • spring.provides 配置所依賴的artifactId惭蟋,給IDE使用的苗桂,沒有其他的作用。

注意一下告组,沒有一行代碼煤伟。

我們重點(diǎn)看一下pom.xml,因?yàn)檫@個(gè)jar包里面除了這個(gè)沒有啥重要的信息

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
  <modelVersion>4.0.0</modelVersion>
  <parent>
    <groupId>org.mybatis.spring.boot</groupId>
    <artifactId>mybatis-spring-boot</artifactId>
    <version>1.3.1</version>
  </parent>
  <artifactId>mybatis-spring-boot-starter</artifactId>
  <name>mybatis-spring-boot-starter</name>
  <dependencies>
    <dependency>
      <groupId>org.springframework.boot</groupId>
      <artifactId>spring-boot-starter</artifactId>
    </dependency>
    <dependency>
      <groupId>org.springframework.boot</groupId>
      <artifactId>spring-boot-starter-jdbc</artifactId>
    </dependency>
    <dependency>
      <groupId>org.mybatis.spring.boot</groupId>
      <artifactId>mybatis-spring-boot-autoconfigure</artifactId>
    </dependency>
    <dependency>
      <groupId>org.mybatis</groupId>
      <artifactId>mybatis</artifactId>
    </dependency>
    <dependency>
      <groupId>org.mybatis</groupId>
      <artifactId>mybatis-spring</artifactId>
    </dependency>
  </dependencies>
</project>

從上面可以看出,pom.xml文件中會(huì)引入一些jar包便锨,其中除了引入spring-boot-starter围辙,之外重點(diǎn)看一下:mybatis-spring-boot-autoconfigure

我們找到mybatis-spring-boot-autoconfigure.jar文件放案,打開這個(gè)文件酌畜。

圖片

里面包含如下文件:

  • pom.properties 配置maven所需的項(xiàng)目version、groupId和artifactId
  • pom.xml 配置所依賴的jar包
  • additional-spring-configuration-metadata.json 手動(dòng)添加IDE提示功能
  • MANIFEST.MF 這個(gè)文件描述了該Jar文件的很多信息
  • spring.factories SPI會(huì)讀取的文件
  • spring-configuration-metadata.json 系統(tǒng)自動(dòng)生成的IDE提示功能
  • ConfigurationCustomizer 自定義Configuration回調(diào)接口
  • MybatisAutoConfiguration mybatis配置類
  • MybatisProperties mybatis屬性類
  • SpringBootVFS 掃描嵌套的jar包中的類

spring-configuration-metadata.jsonadditional-spring-configuration-metadata.json的功能差不多卿叽,我們?cè)?code>applicationContext.properties文件中輸入spring時(shí),會(huì)自動(dòng)出現(xiàn)下面的配置信息可供選擇恳守,就是這個(gè)功能了考婴。

圖片

來自靈魂的一問:這兩個(gè)文件有什么區(qū)別?

答:如果pom.xml中引入了spring-boot-configuration-processor包,則會(huì)自動(dòng)生成spring-configuration-metadata.json催烘。

如果需要手動(dòng)修改里面的元數(shù)據(jù)沥阱,則可以在additional-spring-configuration-metadata.json中編輯,最終兩個(gè)文件中的元數(shù)據(jù)會(huì)合并到一起伊群。

MybatisProperties類是屬性實(shí)體類:

@ConfigurationProperties(prefix = MybatisProperties.MYBATIS_PREFIX)
public class MybatisProperties {

  public static final String MYBATIS_PREFIX = "mybatis";
  private String configLocation;
  private String[] mapperLocations;
  private String typeAliasesPackage;
  private String typeHandlersPackage;
  private boolean checkConfigLocation = false;
  private ExecutorType executorType;
  private Properties configurationProperties;
  @NestedConfigurationProperty
  private Configuration configuration;

  public String getConfigLocation() {
    return this.configLocation;
  }

  public void setConfigLocation(String configLocation) {
    this.configLocation = configLocation;
  }

  @Deprecated
  public String getConfig() {
    return this.configLocation;
  }

  @Deprecated
  public void setConfig(String config) {
    this.configLocation = config;
  }

  public String[] getMapperLocations() {
    return this.mapperLocations;
  }

  public void setMapperLocations(String[] mapperLocations) {
    this.mapperLocations = mapperLocations;
  }
  
  public String getTypeHandlersPackage() {
    return this.typeHandlersPackage;
  }

  public void setTypeHandlersPackage(String typeHandlersPackage) {
    this.typeHandlersPackage = typeHandlersPackage;
  }

  public String getTypeAliasesPackage() {
    return this.typeAliasesPackage;
  }

  public void setTypeAliasesPackage(String typeAliasesPackage) {
    this.typeAliasesPackage = typeAliasesPackage;
  }

  public boolean isCheckConfigLocation() {
    return this.checkConfigLocation;
  }

  public void setCheckConfigLocation(boolean checkConfigLocation) {
    this.checkConfigLocation = checkConfigLocation;
  }

  public ExecutorType getExecutorType() {
    return this.executorType;
  }

  public void setExecutorType(ExecutorType executorType) {
    this.executorType = executorType;
  }

  public Properties getConfigurationProperties() {
    return configurationProperties;
  }

  public void setConfigurationProperties(Properties configurationProperties) {
    this.configurationProperties = configurationProperties;
  }

  public Configuration getConfiguration() {
    return configuration;
  }

  public void setConfiguration(Configuration configuration) {
    this.configuration = configuration;
  }

  public Resource[] resolveMapperLocations() {
    ResourcePatternResolver resourceResolver = new PathMatchingResourcePatternResolver();
    List<Resource> resources = new ArrayList<Resource>();
    if (this.mapperLocations != null) {
      for (String mapperLocation : this.mapperLocations) {
        try {
          Resource[] mappers = resourceResolver.getResources(mapperLocation);
          resources.addAll(Arrays.asList(mappers));
        } catch (IOException e) {
          // ignore
        }
      }
    }
    return resources.toArray(new Resource[resources.size()]);
  }
}

可以看到Mybatis初始化所需要的很多屬性都在這里考杉,相當(dāng)于一個(gè)JavaBean

下面重點(diǎn)看一下MybatisAutoConfiguration的代碼:

@org.springframework.context.annotation.Configuration
@ConditionalOnClass({ SqlSessionFactory.class, SqlSessionFactoryBean.class })
@ConditionalOnBean(DataSource.class)
@EnableConfigurationProperties(MybatisProperties.class)
@AutoConfigureAfter(DataSourceAutoConfiguration.class)
public class MybatisAutoConfiguration {

  private static final Logger logger = LoggerFactory.getLogger(MybatisAutoConfiguration.class);
  private final MybatisProperties properties;
  private final Interceptor[] interceptors;
  private final ResourceLoader resourceLoader;
  private final DatabaseIdProvider databaseIdProvider;
  private final List<ConfigurationCustomizer> configurationCustomizers;
  public MybatisAutoConfiguration(MybatisProperties properties,
                                  ObjectProvider<Interceptor[]> interceptorsProvider,
                                  ResourceLoader resourceLoader,
                                  ObjectProvider<DatabaseIdProvider> databaseIdProvider,
                                  ObjectProvider<List<ConfigurationCustomizer>> configurationCustomizersProvider) {
    this.properties = properties;
    this.interceptors = interceptorsProvider.getIfAvailable();
    this.resourceLoader = resourceLoader;
    this.databaseIdProvider = databaseIdProvider.getIfAvailable();
    this.configurationCustomizers = configurationCustomizersProvider.getIfAvailable();
  }

  @PostConstruct
  public void checkConfigFileExists() {
    if (this.properties.isCheckConfigLocation() && StringUtils.hasText(this.properties.getConfigLocation())) {
      Resource resource = this.resourceLoader.getResource(this.properties.getConfigLocation());
      Assert.state(resource.exists(), "Cannot find config location: " + resource
          + " (please add config file or check your Mybatis configuration)");
    }
  }

  @Bean
  @ConditionalOnMissingBean
  public SqlSessionFactory sqlSessionFactory(DataSource dataSource) throws Exception {
    SqlSessionFactoryBean factory = new SqlSessionFactoryBean();
    factory.setDataSource(dataSource);
    factory.setVfs(SpringBootVFS.class);
    if (StringUtils.hasText(this.properties.getConfigLocation())) {
      factory.setConfigLocation(this.resourceLoader.getResource(this.properties.getConfigLocation()));
    }
    Configuration configuration = this.properties.getConfiguration();
    if (configuration == null && !StringUtils.hasText(this.properties.getConfigLocation())) {
      configuration = new Configuration();
    }
    if (configuration != null && !CollectionUtils.isEmpty(this.configurationCustomizers)) {
      for (ConfigurationCustomizer customizer : this.configurationCustomizers) {
        customizer.customize(configuration);
      }
    }
    factory.setConfiguration(configuration);
    if (this.properties.getConfigurationProperties() != null) {
      factory.setConfigurationProperties(this.properties.getConfigurationProperties());
    }
    if (!ObjectUtils.isEmpty(this.interceptors)) {
      factory.setPlugins(this.interceptors);
    }
    if (this.databaseIdProvider != null) {
      factory.setDatabaseIdProvider(this.databaseIdProvider);
    }
    if (StringUtils.hasLength(this.properties.getTypeAliasesPackage())) {
      factory.setTypeAliasesPackage(this.properties.getTypeAliasesPackage());
    }
    if (StringUtils.hasLength(this.properties.getTypeHandlersPackage())) {
      factory.setTypeHandlersPackage(this.properties.getTypeHandlersPackage());
    }
    if (!ObjectUtils.isEmpty(this.properties.resolveMapperLocations())) {
      factory.setMapperLocations(this.properties.resolveMapperLocations());
    }

    return factory.getObject();
  }

  @Bean
  @ConditionalOnMissingBean
  public SqlSessionTemplate sqlSessionTemplate(SqlSessionFactory sqlSessionFactory) {
    ExecutorType executorType = this.properties.getExecutorType();
    if (executorType != null) {
      return new SqlSessionTemplate(sqlSessionFactory, executorType);
    } else {
      return new SqlSessionTemplate(sqlSessionFactory);
    }
  }

  public static class AutoConfiguredMapperScannerRegistrar
      implements BeanFactoryAware, ImportBeanDefinitionRegistrar, ResourceLoaderAware {
    private BeanFactory beanFactory;
    private ResourceLoader resourceLoader;

    @Override
    public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {

      ClassPathMapperScanner scanner = new ClassPathMapperScanner(registry);
      try {
        if (this.resourceLoader != null) {
          scanner.setResourceLoader(this.resourceLoader);
        }

        List<String> packages = AutoConfigurationPackages.get(this.beanFactory);
        if (logger.isDebugEnabled()) {
          for (String pkg : packages) {
            logger.debug("Using auto-configuration base package '{}'", pkg);
          }
        }

        scanner.setAnnotationClass(Mapper.class);
        scanner.registerFilters();
        scanner.doScan(StringUtils.toStringArray(packages));
      } catch (IllegalStateException ex) {
        logger.debug("Could not determine auto-configuration package, automatic mapper scanning disabled.", ex);
      }
    }

    @Override
    public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
      this.beanFactory = beanFactory;
    }

    @Override
    public void setResourceLoader(ResourceLoader resourceLoader) {
      this.resourceLoader = resourceLoader;
    }
  }

  @org.springframework.context.annotation.Configuration
  @Import({ AutoConfiguredMapperScannerRegistrar.class })
  @ConditionalOnMissingBean(MapperFactoryBean.class)
  public static class MapperScannerRegistrarNotFoundConfiguration {
  
    @PostConstruct
    public void afterPropertiesSet() {
      logger.debug("No {} found.", MapperFactoryBean.class.getName());
    }
  }
}

這個(gè)類就是一個(gè)Configuration(配置類)舰始,它里面定義很多bean崇棠,其中最重要的就是SqlSessionFactory的bean實(shí)例,該實(shí)例是Mybatis的核心功能丸卷,用它創(chuàng)建SqlSession枕稀,對(duì)數(shù)據(jù)庫進(jìn)行CRUD操作。

除此之外谜嫉,MybatisAutoConfiguration類還包含了:

  • @ConditionalOnClass 配置了只有包含SqlSessionFactory.class和SqlSessionFactoryBean.class萎坷,該配置類才生效。
  • @ConditionalOnBean 配置了只有包含dataSource實(shí)例時(shí)沐兰,該配置類才生效哆档。
  • @EnableConfigurationProperties 該注解會(huì)自動(dòng)填充MybatisProperties實(shí)例中的屬性。
  • AutoConfigureAfter 配置了該配置類在DataSourceAutoConfiguration類之后自動(dòng)配置住闯。

這些注解都是一些輔助功能瓜浸,決定Configuration是否生效,當(dāng)然這些注解不是必須的比原。

接下來斟叼,重點(diǎn)看看spring.factories文件有啥內(nèi)容:

org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
org.mybatis.spring.boot.autoconfigure.MybatisAutoConfiguration

里面只有一行配置,即keyEnableAutoConfiguration春寿,valueMybatisAutoConfiguration朗涩。

好了,介紹了這么多東西绑改,現(xiàn)在我們來總結(jié)一下谢床,

starter幾個(gè)要素如下圖所示:
圖片

那么兄一,編寫starter需要哪些步驟?

  • 1.需要定義一個(gè)名稱為xxx-spring-boot-starter的空項(xiàng)目,里面不包含任何代碼识腿,可以有pom.xml和pom.properties文件出革。
  • 2.pom.xml文件中包含了名稱為xxx-spring-boot-autoconfigure的項(xiàng)目。
  • 3.xxx-spring-boot-autoconfigure項(xiàng)目中包含了名稱為xxxAutoConfiguration的類渡讼,該類可以定義一些bean實(shí)例骂束。當(dāng)然,Configuration類上可以打一些如:ConditionalOnClass成箫、ConditionalOnBean展箱、EnableConfigurationProperties等注解。
  • 4.需要在spring.factories文件中增加key為EnableAutoConfiguration蹬昌,value為xxxAutoConfiguration混驰。

我們?cè)囍凑者@四步,自己編寫一個(gè)starter看看能否成功皂贩,驗(yàn)證一下總結(jié)的內(nèi)容是否正確栖榨。

3 如何定義自己的starter?

3.1 先創(chuàng)建一個(gè)空項(xiàng)目

該項(xiàng)目名稱為id-generate-starter明刷,注意為了方便我把項(xiàng)目重命名了婴栽,原本應(yīng)該是叫id-generate-spring-boot-starter的,如下圖所示:

圖片

pom.xml文件定義如下:

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <version>1.3.1</version>
    <groupId>com.sue</groupId>
    <artifactId>id-generate-spring-boot-starter</artifactId>
    <name>id-generate-spring-boot-starter</name>
    <dependencies>
        <dependency>
            <groupId>com.sue</groupId>
            <artifactId>id-generate-spring-boot-autoconfigure</artifactId>
            <version>1.3.1</version>
        </dependency>
    </dependencies>
</project>

我們看到辈末,它只引入了id-generate-spring-boot-autoconfigure居夹。當(dāng)然如果有需要這里還可以引入多個(gè)autoconfigure或者多個(gè)其他jar包或者。

3.2 創(chuàng)建id-generate-autoconfigure

同樣為了方便我把項(xiàng)目重命名了本冲,原本是叫id-generate-spring-boot-autoconfigure准脂,如下圖所示:

圖片

該項(xiàng)目當(dāng)中包含:pom.xml、spring.factories檬洞、IdGenerateAutoConfiguration狸膏、IdGenerateService 和 IdProperties 這5個(gè)關(guān)鍵文件,下面我們逐一看看添怔。

先從pom.xml

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.0.4.RELEASE</version>
    </parent>
    <modelVersion>4.0.0</modelVersion>
    <version>1.3.1</version>
    <groupId>com.sue</groupId>
    <artifactId>id-generate-spring-boot-autoconfigure</artifactId>
    <name>id-generate-spring-boot-autoconfigure</name>

    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-autoconfigure</artifactId>
        </dependency>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-configuration-processor</artifactId>
            <optional>true</optional>
        </dependency>
    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-compiler-plugin</artifactId>
                <configuration>
                    <source>1.8</source>
                    <target>1.8</target>
                </configuration>
            </plugin>
        </plugins>
    </build>
</project>

我們可以看到湾戳,這個(gè)文件比較簡(jiǎn)單就引入了:

  • spring-boot-starter:springboot的相關(guān)jar包。
  • spring-boot-autoconfigure:springboot自動(dòng)配置相關(guān)jar包广料。
  • spring-boot-configuration-processor:springboot生成IDE提示功能相關(guān)jar包砾脑。

重點(diǎn)看看spring.factories文件:

org.springframework.boot.autoconfigure.EnableAutoConfiguration=com.sue.IdGenerateAutoConfiguration

它里面只包含一行配置,其中key是EnableAutoConfiguration艾杏,value是IdGenerateAutoConfiguration韧衣。

再重點(diǎn)看一下IdGenerateAutoConfiguration

@ConditionalOnClass(IdProperties.class)
@EnableConfigurationProperties(IdProperties.class)
@Configuration
public class IdGenerateAutoConfiguration {
    @Autowired
    private IdProperties properties;
    @Bean
    public IdGenerateService idGenerateService() {
        return new IdGenerateService(properties.getWorkId());
    }
}

該類是一個(gè)使用了@Configuration注解標(biāo)記為了配置類,生效的條件是@ConditionalOnClass注解中檢測(cè)到包含IdProperties.class。并且使用@EnableConfigurationProperties注解會(huì)自動(dòng)注入IdProperties的實(shí)例畅铭。

此外氏淑,最關(guān)鍵的點(diǎn)是該類里面創(chuàng)建了idGenerateService的bean實(shí)例,這是自動(dòng)配置的精髓硕噩。

再看看IdGenerateService

public class IdGenerateService {
    private Long workId;
    public IdGenerateService(Long workId) {
        this.workId = workId;
    }

    public Long generate() {
        return new Random().nextInt(100) + this.workId;
    }
}

我們可以看到它是一個(gè)普通的類假残,甚至都沒有使用@Service注解,里面有個(gè)generate方法炉擅,根據(jù)workId的值和隨機(jī)數(shù)動(dòng)態(tài)生成id辉懒。

最后看看IdProperties

@ConfigurationProperties(prefix = IdProperties.PREFIX)
public class IdProperties {
    public static final String PREFIX = "sue";
    private Long workId;
    public Long getWorkId() {
        return workId;
    }
    public void setWorkId(Long workId) {
        this.workId = workId;
    }
}

它是一個(gè)配置實(shí)體類,里面包含了相關(guān)的配置文件谍失。使用@ConfigurationProperties注解眶俩,會(huì)自動(dòng)把application.properties文件中以sue開通的,參數(shù)名稱跟IdProperties中一樣的參數(shù)值袱贮,自動(dòng)注入到IdProperties對(duì)象中。

3.3 創(chuàng)建id-generate-test

這個(gè)項(xiàng)目主要用于測(cè)試体啰。

圖片

該項(xiàng)目里面包含:pom.xml攒巍、application.properties、Application 和 TestRunner 文件荒勇。

先看看pom.xml文件

<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">

    <modelVersion>4.0.0</modelVersion>
    <version>1.3.1</version>
    <groupId>com.sue</groupId>
    <artifactId>spring-boot-id-generate-test</artifactId>
    <name>spring-boot-id-generate-test</name>
    <dependencies>
        <dependency>
            <groupId>com.sue</groupId>
            <artifactId>id-generate-spring-boot-starter</artifactId>
            <version>1.3.1</version>
        </dependency>
    </dependencies>
</project>

由于只測(cè)試剛剛定義的id生成功能柒莉,所以只引入的id-generate-spring-boot-starter jar包。

application.properties配置資源文件

sue.workId=123

只有一行配置沽翔,因?yàn)槲覀兊腎dProperties中目前只需要這一個(gè)參數(shù)兢孝。

Application是測(cè)試程序啟動(dòng)類

@SpringBootApplication
public class Application {
    public static void main(String[] args) {
        SpringApplication.run(Application.class, args);
    }
}

很簡(jiǎn)單,就是一個(gè)普通的springboot啟動(dòng)類

TestRunner是我們的測(cè)試類

@Component
public class TestRunner implements ApplicationRunner {
    @Autowired
    private IdGenerateService idGenerateService;
    public void run(ApplicationArguments args) throws Exception {
        Long sysNo = idGenerateService.generate();
        System.out.println(sysNo);
    }
}

它實(shí)現(xiàn)了ApplicationRunner接口仅偎,所以在springboot啟動(dòng)的時(shí)候會(huì)調(diào)用該類的run方法跨蟹。

好了,所有自定義starter的代碼和測(cè)試代碼都已經(jīng)就緒橘沥。接下窗轩,運(yùn)行一下Application類的main方法。

運(yùn)行結(jié)果:

176

完美座咆,驗(yàn)證成功了痢艺。

接下來,我們分析一下starter的底層實(shí)現(xiàn)原理介陶。

4 starter的底層原理是什么堤舒?

通過上面編寫自己的starter的例子,相信大家對(duì)starter的認(rèn)識(shí)更進(jìn)一步了哺呜,現(xiàn)在跟大家一起看看starter的底層是如何實(shí)現(xiàn)的舌缤。

id-generate-starter.jar其實(shí)是一個(gè)空項(xiàng)目,依賴于id-generate-autoconfiguration.jar。

id-generate-starter.jar是一個(gè)入口友驮,我們給他取一個(gè)更優(yōu)雅的名字:門面模式漂羊,其他業(yè)務(wù)系統(tǒng)想引入相應(yīng)的功能,必須要通過這個(gè)門面卸留。

我們重點(diǎn)分析一下 id-generate-autoconfiguration.jar

該jar包核心內(nèi)容是:IdGenerateConfiguration走越,這個(gè)配置類中創(chuàng)建了IdGenerateService對(duì)象,IdGenerateService是我們所需要自動(dòng)配置的具體功能耻瑟。

接下來一個(gè)最重要的問題:IdGenerateConfiguration為什么會(huì)自動(dòng)加載的呢旨指?

還記得我們定義的spring.factories文件不?

org.springframework.boot.autoconfigure.EnableAutoConfiguration=com.sue.IdGenerateAutoConfiguration

它里面只包含一行配置喳整,其中keyEnableAutoConfiguration谆构,valueIdGenerateAutoConfiguration

要搞明白這個(gè)過程框都,要從Application類的@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 {

  @AliasFor(annotation = EnableAutoConfiguration.class)
  Class<?>[] exclude() default {};

  @AliasFor(annotation = EnableAutoConfiguration.class)
  String[] excludeName() default {};

  @AliasFor(annotation = ComponentScan.class, attribute = "basePackages")
  String[] scanBasePackages() default {};
  
  @AliasFor(annotation = ComponentScan.class, attribute = "basePackageClasses")
  Class<?>[] scanBasePackageClasses() default {};
}

從上面可以看出該注解里面包含了@EnableAutoConfiguration注解搬素。

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@AutoConfigurationPackage
@Import(AutoConfigurationImportSelector.class)
public @interface EnableAutoConfiguration {
  String ENABLED_OVERRIDE_PROPERTY = "spring.boot.enableautoconfiguration";

  Class<?>[] exclude() default {};
  String[] excludeName() default {};
}

@EnableAutoConfiguration注解會(huì)引入AutoConfigurationImportSelector類。

該類的selectImports方法一個(gè)關(guān)鍵方法:

@Override
  public String[] selectImports(AnnotationMetadata annotationMetadata) {
    //配置有沒有配置spring.boot.enableautoconfiguration開關(guān)魏保,默認(rèn)為true
    //如果為false熬尺,則不執(zhí)行自動(dòng)配置的功能,直接返回
    if (!isEnabled(annotationMetadata)) {
      return NO_IMPORTS;
    }
    //找spring-autoconfigure-metadata.properties中的元素
    AutoConfigurationMetadata autoConfigurationMetadata = AutoConfigurationMetadataLoader
        .loadMetadata(this.beanClassLoader);
    //獲取EnableAutoConfiguration注解中的屬性 
    AnnotationAttributes attributes = getAttributes(annotationMetadata);
    //獲取工程下所有配置key為EnableAutoConfiguration的值谓罗,即IdGenerateConfiguration等類粱哼。
    List<String> configurations = getCandidateConfigurations(annotationMetadata,
        attributes);
    //刪除重復(fù)的值    
    configurations = removeDuplicates(configurations);
    //獲取需要排除的規(guī)則列表
    Set<String> exclusions = getExclusions(annotationMetadata, attributes);
    //檢查
    checkExcludedClasses(configurations, exclusions);
    //刪除需要排除的值
    configurations.removeAll(exclusions);
    //根據(jù)配置文件中配置的開關(guān),過濾一部分不滿足條件的值
    configurations = filter(configurations, autoConfigurationMetadata);
    fireAutoConfigurationImportEvents(configurations, exclusions);
    return StringUtils.toStringArray(configurations);
  }

這里就是starter能夠自動(dòng)配置的秘密檩咱。

此外揭措,有些朋友看其他人定義的springboot starter可能會(huì)有疑惑。

先看看druid-spring-boot-starter

圖片

alibaba定義的druid-spring-boot-starter只有xxx-spring-boot-starter.jar文件刻蚯,而沒有xxx-spring-boot-autoconfigure.jar文件绊含。

再看看spring-boot-starter-jdbc

圖片

更神奇的是這個(gè)文件中連pom.xml都沒有,一臉懵逼炊汹。艺挪。。兵扬。麻裳。。器钟。

是不是我講錯(cuò)了津坑?

答:其實(shí)沒有。

SpringBoot的原則是約定優(yōu)于配置傲霸。

從spring-boot-starter-jdbc內(nèi)部空實(shí)現(xiàn)來看疆瑰,它的約定是要把xxx-spring-boot-starter.jar和xxx-spring-boot-autoconfigure.jar區(qū)分開的眉反。個(gè)人認(rèn)為,alibaba定義得并不好穆役,沒有遵照springboot的約定寸五,雖然功能不受影響。(這個(gè)地方歡迎一起探討一下)

而springboot自己定義的spring-boot-starter-jdbc為什么連pom.xml文件也沒有呢耿币?

它不需要依賴xxx-spring-boot-autoconfigure.jar文件嗎梳杏?

因?yàn)閟pringboot把所有的自動(dòng)配置的類都統(tǒng)一放到spring-boot-autoconfigure.jar下面了:

圖片

spring.factories文件內(nèi)容如下:

圖片

SpringBoot這樣集中管理自動(dòng)配置,而不需要從各個(gè)子包中遍歷淹接,我個(gè)人認(rèn)為是為了查找效率十性。

我們最后再看看spring-cloud-starter-openfegin

圖片

明顯看到,它是遵循了我們說的原則的塑悼。

除此之外劲适,還有一個(gè)原則一順便提一下。

SpringBootSpringCloud系列定義jar包的名稱是:

  • spring-boot-starter-xxx.jar
  • spring-cloud-starter-xxx.jar

而我們自己的項(xiàng)目定義的jar應(yīng)該是:

  • xxx-spring-boot-starter.jar
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末厢蒜,一起剝皮案震驚了整個(gè)濱河市霞势,隨后出現(xiàn)的幾起案子贰盗,更是在濱河造成了極大的恐慌铛漓,老刑警劉巖,帶你破解...
    沈念sama閱讀 222,000評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件暇矫,死亡現(xiàn)場(chǎng)離奇詭異鄙才,居然都是意外死亡颂鸿,警方通過查閱死者的電腦和手機(jī)促绵,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,745評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門攒庵,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人败晴,你說我怎么就攤上這事浓冒。” “怎么了尖坤?”我有些...
    開封第一講書人閱讀 168,561評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵稳懒,是天一觀的道長。 經(jīng)常有香客問我慢味,道長场梆,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 59,782評(píng)論 1 298
  • 正文 為了忘掉前任纯路,我火速辦了婚禮或油,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘驰唬。我一直安慰自己顶岸,他們只是感情好腔彰,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,798評(píng)論 6 397
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著辖佣,像睡著了一般霹抛。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上卷谈,一...
    開封第一講書人閱讀 52,394評(píng)論 1 310
  • 那天杯拐,我揣著相機(jī)與錄音,去河邊找鬼雏搂。 笑死藕施,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的凸郑。 我是一名探鬼主播裳食,決...
    沈念sama閱讀 40,952評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼芙沥!你這毒婦竟也來了诲祸?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,852評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤而昨,失蹤者是張志新(化名)和其女友劉穎救氯,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體歌憨,經(jīng)...
    沈念sama閱讀 46,409評(píng)論 1 318
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡着憨,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,483評(píng)論 3 341
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了务嫡。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片甲抖。...
    茶點(diǎn)故事閱讀 40,615評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖心铃,靈堂內(nèi)的尸體忽然破棺而出准谚,到底是詐尸還是另有隱情,我是刑警寧澤去扣,帶...
    沈念sama閱讀 36,303評(píng)論 5 350
  • 正文 年R本政府宣布柱衔,位于F島的核電站,受9級(jí)特大地震影響愉棱,放射性物質(zhì)發(fā)生泄漏唆铐。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,979評(píng)論 3 334
  • 文/蒙蒙 一奔滑、第九天 我趴在偏房一處隱蔽的房頂上張望艾岂。 院中可真熱鬧,春花似錦档押、人聲如沸澳盐。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,470評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽叼耙。三九已至腕窥,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間筛婉,已是汗流浹背簇爆。 一陣腳步聲響...
    開封第一講書人閱讀 33,571評(píng)論 1 272
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留爽撒,地道東北人入蛆。 一個(gè)月前我還...
    沈念sama閱讀 49,041評(píng)論 3 377
  • 正文 我出身青樓,卻偏偏與公主長得像硕勿,于是被迫代替她去往敵國和親哨毁。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,630評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容