在咱们学习SpringBoot
时都已经了解到starter
是SpringBoot
的核心组成部分,SpringBoot
为咱们提供了尽量完善的封装,提供了一系列的自动化配置的starter
插件,咱们在使用spring-boot-starter-web
时只须要在pom.xml
配置文件内添加依赖就能够了,咱们以前传统方式则是须要添加不少相关SpringMVC
配置文件。而spring-boot-starter-web
为咱们提供了几乎全部的默认配置,很好的下降了使用框架时的复杂度。
所以在使用xx.starter
时你就不用考虑该怎么配置,即使是有一些必要的配置在application.properties
配置文件内对应配置就能够了,那好,为何我在application.properties
配置对应属性后xx.starter
就能够获取到并做出处理呢?下面咱们带着这个疑问来编写咱们自定义的starter
让咱们深刻了解SpringBoot
java
自定义starter
而且经过spring-boot-autoconfigure
完成自动化配置。git
建立starter
项目咱们并不须要建立SpringBoot
项目,咱们建立一个Maven
项目就能够知足咱们的需求,建立项目完成后pom.xml
配置信息以下所示:web
<?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>
<groupId>com.yuqiyu</groupId>
<artifactId>chapter28</artifactId>
<version>1.0.0</version>
<packaging>jar</packaging>
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
</properties>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-autoconfigure</artifactId>
<version>1.5.4.RELEASE</version>
</dependency>
</dependencies>
</project>复制代码
咱们这个starter
并不作其余复杂逻辑的编写,因此这里的依赖只是添加了spring-boot-autoconfigure
,实战开发时能够添加任意依赖到项目中。spring
咱们在文章开头埋下了一个疑问,starter
是如何读取application.properties
或者application.yml
配置文件内须要的配置参数的呢?那么接下来咱们就看看如何能够获取自定义的配置信息。SpringBoot
在处理这种事情上早就已经考虑到了,因此提供了一个注解@ConfigurationProperties
,该注解能够完成将application.properties
配置文件内的有规则的配置参数映射到实体内的field
内,不过须要提供setter方法,自定义配置参数实体代码以下所示:apache
package com.yuqiyu.chapter28;
import org.springframework.boot.context.properties.ConfigurationProperties;
/**
* 配置文件实体映射
* ========================
* Created with IntelliJ IDEA.
* User:恒宇少年
* Date:2017/7/22
* Time:22:51
* 码云:http://git.oschina.net/jnyqy
* ========================
*/
@ConfigurationProperties(prefix = "hello")
public class HelloProperties
{
//消息内容
private String msg = "HengYu";
//是否显示消息内容
private boolean show = true;
public String getMsg() {
return msg;
}
public void setMsg(String msg) {
this.msg = msg;
}
public boolean isShow() {
return show;
}
public void setShow(boolean show) {
this.show = show;
}
}复制代码
在上面代码中,@ConfigurationProperties
注解内咱们使用到了属性preffix
,该属性配置了读取参数的前缀,根据上面的实体属性对应配置文件内的配置则是hello.msg
、hello.show
,固然咱们提供了默认值,配置文件内不进行配置时则是使用默认值。springboot
咱们为自定义starter
提供一个Service
,而且提供一个名为sayHello
的方法用于返回咱们配置的msg
内容。代码以下所示:bash
package com.yuqiyu.chapter28;
/**
* 自定义业务实现
* ========================
* Created with IntelliJ IDEA.
* User:恒宇少年
* Date:2017/7/22
* Time:22:54
* 码云:http://git.oschina.net/jnyqy
* ========================
*/
public class HelloService
{
//消息内容
private String msg;
//是否显示消息内容
private boolean show = true;
public String sayHello()
{
return show ? "Hello," + msg : "Hidden";
}
public void setMsg(String msg) {
this.msg = msg;
}
public void setShow(boolean show) {
this.show = show;
}
}复制代码
咱们Service
内的代码比较简单,根据属性参数进行返回格式化后的字符串。app
接下来咱们开始编写自动配置,这一块是starter
的核心部分,配置该部分后在启动项目时才会自动加载配置,固然其中有不少细节性质的配置框架
自动化配置其实只是提供实体bean的验证以及初始化,咱们先来看看代码:maven
package com.yuqiyu.chapter28;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass;
import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean;
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
import org.springframework.boot.context.properties.EnableConfigurationProperties;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
/**
* 自定义starter自动化配置
* ========================
* Created with IntelliJ IDEA.
* User:恒宇少年
* Date:2017/7/22
* Time:22:56
* 码云:http://git.oschina.net/jnyqy
* ========================
*/
@Configuration//开启配置
@EnableConfigurationProperties(HelloProperties.class)//开启使用映射实体对象
@ConditionalOnClass(HelloService.class)//存在HelloService时初始化该配置类
@ConditionalOnProperty//存在对应配置信息时初始化该配置类
(
prefix = "hello",//存在配置前缀hello
value = "enabled",//开启
matchIfMissing = true//缺失检查
)
public class HelloAutoConfiguration
{
//application.properties配置文件映射前缀实体对象
@Autowired
private HelloProperties helloProperties;
/**
* 根据条件判断不存在HelloService时初始化新bean到SpringIoc
* @return
*/
@Bean//建立HelloService实体bean
@ConditionalOnMissingBean(HelloService.class)//缺失HelloService实体bean时,初始化HelloService并添加到SpringIoc
public HelloService helloService()
{
System.out.println(">>>The HelloService Not Found,Execute Create New Bean.");
HelloService helloService = new HelloService();
helloService.setMsg(helloProperties.getMsg());//设置消息内容
helloService.setShow(helloProperties.isShow());//设置是否显示
return helloService;
}
}复制代码
自动化配置代码中有不少咱们以前没有用到的注解配置,咱们从上开始讲解
@Configuration
:这个配置就不用多作解释了,咱们一直在使用@EnableConfigurationProperties
:这是一个开启使用配置参数的注解,value
值就是咱们配置实体参数映射的ClassType
,将配置实体做为配置来源。
有关@ConditionalOnXxx
相关的注解这里要系统的说下,由于这个是咱们配置的关键,根据名称咱们能够理解为具备Xxx条件
,固然它实际的意义也是如此,条件注解是一个系列,下面咱们详细作出解释
@ConditionalOnBean
:当SpringIoc
容器内存在指定Bean
的条件@ConditionalOnClass
:当SpringIoc
容器内存在指定Class
的条件@ConditionalOnExpression
:基于SpEL表达式做为判断条件@ConditionalOnJava
:基于JVM
版本做为判断条件@ConditionalOnJndi
:在JNDI存在时查找指定的位置@ConditionalOnMissingBean
:当SpringIoc
容器内不存在指定Bean
的条件@ConditionalOnMissingClass
:当SpringIoc
容器内不存在指定Class
的条件@ConditionalOnNotWebApplication
:当前项目不是Web项目的条件@ConditionalOnProperty
:指定的属性是否有指定的值@ConditionalOnResource
:类路径是否有指定的值@ConditionalOnSingleCandidate
:当指定Bean
在SpringIoc
容器内只有一个,或者虽然有多个可是指定首选的Bean
@ConditionalOnWebApplication
:当前项目是Web项目的条件
以上注解都是元注解@Conditional
演变而来的,根据不用的条件对应建立以上的具体条件注解。
到目前为止咱们尚未完成自动化配置starter
,咱们须要了解SpringBoot
运做原理后才能够完成后续编码。
在注解@SpringBootApplication
上存在一个开启自动化配置的注解@EnableAutoConfiguration
来完成自动化配置,注解源码以下所示:
//
// Source code recreated from a .class file by IntelliJ IDEA
// (powered by Fernflower decompiler)
//
package org.springframework.boot.autoconfigure;
import java.lang.annotation.Documented;
import java.lang.annotation.ElementType;
import java.lang.annotation.Inherited;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
import org.springframework.context.annotation.Import;
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
@AutoConfigurationPackage
@Import({EnableAutoConfigurationImportSelector.class})
public @interface EnableAutoConfiguration {
String ENABLED_OVERRIDE_PROPERTY = "spring.boot.enableautoconfiguration";
Class<?>[] exclude() default {};
String[] excludeName() default {};
}复制代码
在@EnableAutoConfiguration
注解内使用到了@import
注解来完成导入配置的功能,而EnableAutoConfigurationImportSelector
内部则是使用了SpringFactoriesLoader.loadFactoryNames
方法进行扫描具备META-INF/spring.factories
文件的jar包。咱们能够先来看下spring-boot-autoconfigure
包内的spring.factories
文件内容,以下所示:
# Initializers
org.springframework.context.ApplicationContextInitializer=\
org.springframework.boot.autoconfigure.SharedMetadataReaderFactoryContextInitializer,\
org.springframework.boot.autoconfigure.logging.AutoConfigurationReportLoggingInitializer
# Application Listeners
org.springframework.context.ApplicationListener=\
org.springframework.boot.autoconfigure.BackgroundPreinitializer
# Auto Configuration Import Listeners
org.springframework.boot.autoconfigure.AutoConfigurationImportListener=\
org.springframework.boot.autoconfigure.condition.ConditionEvaluationReportAutoConfigurationImportListener
# Auto Configuration Import Filters
org.springframework.boot.autoconfigure.AutoConfigurationImportFilter=\
org.springframework.boot.autoconfigure.condition.OnClassCondition
# Auto Configure
org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
org.springframework.boot.autoconfigure.admin.SpringApplicationAdminJmxAutoConfiguration,\
org.springframework.boot.autoconfigure.aop.AopAutoConfiguration,\
.....省略复制代码
能够看到配置的结构形式是Key
=>Value
形式,多个Value
时使用,
隔开,那咱们在自定义starter
内也可使用这种形式来完成,咱们的目的是为了完成自动化配置,因此咱们这里Key
则是须要使用org.springframework.boot.autoconfigure.EnableAutoConfiguration
咱们在src/main/resource
目录下建立META-INF
目录,并在目录内添加文件spring.factories
,具体内容以下所示:
#配置自定义Starter的自动化配置
org.springframework.boot.autoconfigure.EnableAutoConfiguration=com.yuqiyu.chapter28.HelloAutoConfiguration复制代码
都目前为止咱们的自定义starter
已经配置完成,下面咱们须要新建一个SpringBoot
项目来测试咱们的自动化配置是否已经生效。
在使用自定义starter
以前须要将starter
做Maven Jar Install
到本地,咱们使用idea工具自带的maven命令完成该操做
步骤:工具右侧 -> Maven Projects -> Lifecycle -> install
建立测试项目的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>
<groupId>com.yuqiyu.sample</groupId>
<artifactId>test-spring-boot-starter-hello</artifactId>
<version>0.0.1-SNAPSHOT</version>
<packaging>jar</packaging>
<name>test-spring-boot-starter-hello</name>
<description>Demo project for Spring Boot</description>
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>1.5.4.RELEASE</version>
<relativePath/> <!-- lookup parent from repository -->
</parent>
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
<project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
<java.version>1.8</java.version>
</properties>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter</artifactId>
</dependency>
<!--自定义starter依赖-->
<dependency>
<groupId>com.yuqiyu</groupId>
<artifactId>chapter28</artifactId>
<version>1.0.0</version>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
</plugins>
</build>
</project>复制代码
咱们只须要将依赖添加到pom.xml
配置文件内
在运行项目以前,咱们打开application.properties
配置文件开启debug
模式,查看自动化配置的输出日志,配置内容以下所示:
#显示debug日志信息
debug=true复制代码
接下来咱们启动项目,在控制台查找是否存在咱们的HelloAutoConfiguration
日志输出,控制台输出内容以下所示:
.....省略
>>>The HelloService Not Found,Execute Create New Bean.
.....省略
HelloAutoConfiguration matched:
- @ConditionalOnClass found required class 'com.yuqiyu.chapter28.HelloService'; @ConditionalOnMissingClass did not find unwanted class (OnClassCondition)
- @ConditionalOnProperty (hello.enabled) matched (OnPropertyCondition)
HelloAutoConfiguration#helloService matched:
- @ConditionalOnMissingBean (types: com.yuqiyu.chapter28.HelloService; SearchStrategy: all) did not find any beans (OnBeanCondition)
.....省略复制代码
在控制台能够看到咱们的自定义starter
的自动化配置已经生效了,而且根据@ConditionalOnMissingBean(HelloService.class)
作出了条件注入HelloService
实体bean到SpringIoc
容器内
咱们来编写一个简单的测试控制器,查看HelloService
在不配置参数状况下输出格式化字符串内容,控制器代码以下所示:
package com.yuqiyu.sample.testspringbootstarterhello;
import com.yuqiyu.chapter28.HelloService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
/**
* 测试自定义starter自动化配置HelloService
* ========================
* Created with IntelliJ IDEA.
* User:恒宇少年
* Date:2017/7/23
* Time:11:42
* 码云:http://git.oschina.net/jnyqy
* ========================
*/
@RestController
public class HelloController
{
//注入自定义starter内逻辑
@Autowired
HelloService helloService;
/**
* 测试访问地址/hello
* @return 格式化字符串
*/
@RequestMapping(value = "/hello")
public String sayHello()
{
return helloService.sayHello();
}
}复制代码
接下来咱们重启下项目,访问地址http://127.0.0.1:8080/hello,界面输出内容以下所示:
Hello,HengYu复制代码
界面输出的内容是咱们默认值,接下来咱们在application.properties
配置文件内对应添加hello.msg
、hello.show
配置参数,以下所示:
#配置自定义starter参数
hello.msg=HengYu Boy
hello.show=true复制代码
重启项目,再次访问地址,界面输出内容以下所示:
Hello,HengYu Boy复制代码
咱们的配置生效了,到目前为止我相信你们已经明白了咱们application.properties
配置文件为何能够做为统一配置入口,为何配置后能够被对应starter
所使用。
以上内容是本章的所有讲解,本章主要讲解了咱们如何自定义starter
而且自动化配置到SpringBoot
项目中,固然里面还有不少神奇的地方须要你们去深刻挖掘。
本章代码已经上传到码云:
SpringBoot配套源码地址:gitee.com/hengboy/spr…
SpringCloud配套源码地址:gitee.com/hengboy/spr…
SpringBoot相关系列文章请访问:目录:SpringBoot学习目录
QueryDSL相关系列文章请访问:QueryDSL通用查询框架学习目录
SpringDataJPA相关系列文章请访问:目录:SpringDataJPA学习目录
感谢阅读!
欢迎加入QQ技术交流群,共同进步。