Spring Boot精要:
- 自动配置:为常见场景进行自动配置,例如在classpath里发现JdbcTemplate,那么它就自动配置一个JdbcTempllate的bean。
- 起步依赖:特殊的maven或者gradle依赖,利用传递性依赖解析,把常用库聚合在一起,组成了几个为特定功能而定制的依赖。
- Actuator
@SpringBootApplication将三个注解组合在一起
- @Configuration -- 如果有其他配置的话,可以在其他java文件配@Configuration,尽量保持启动类的单一职责
- @ComponentScan
- @EnableAutoConfiguration -- 即自动配置
用gradle运行的2种方式
gradle bootRun /*bootRun任务来自Spring Boot的Gradle插件*/
gradle build
java -jar build/libs/*.jar
在application.yml里可以配置东西覆盖Spring Boot的自动配置,例如修改tomcat启动的端口号:
server:
port: 8081
gradle dependencies -- 分析依赖树
可以在compie("***") {exclude group *} 排除一些不必要的依赖为项目瘦身。
例如, exclude group: 'org.apache.zookeeper', module: 'zookeeper'
因为起步依赖引入的传递依赖都做了很好的兼容性测试,所以在特殊情况下才需要覆盖这些传递依赖,例如新版修复了一个bug。
maven:总会使用路径最近的依赖,可以直接在pom声明一个依赖来覆盖传递依赖。
gradle:倾向于使用库的最新版本,如果要使用新版本覆盖传递依赖里的老版本,直接在配置里声明一个compile("") 就行;如果想用老版本覆盖传递依赖里新版本,则需要:
compile("启动依赖") {exclude 新版本}
comile(老版本)
使用自动配置
例如,Spring的JdbcTemplate是不是在Classpath里,如果是并且有DataSource的Bean,则自动配置一个JdbcTemplate的Bean。
每次应用程序启动的时候,Spring Boot的自动配置都要做将近200个这样的决定,覆盖安全、集成、持久化、web开发等诸多方面。
自动配置利用了spring4.0引入的新特性:条件化配置,它允许配置存在于应用程序中,在满足某些特性条件前都忽略这个配置。引入springboot后,有个名为spring-boot-autoconfigure的JAR文件,包含了很多配置类,每个配置类都在应用程序的classpath下。
在spring里可以很方便的编写你自己的条件,实现Condition接口覆盖matches()方法,举个例子如下,只要在classpath下存在JdbcTemplate才会生效这个条件。
import org.springframework.context.annotation.Condition;
import org.springframework.context.annotation.ConditionContext;
import org.springframework.context.annotation.Conditional;
import org.springframework.core.type.AnnotatedTypeMetadata;
public class MyCondition implements Condition {
@Override
public boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata) {
try {
context.getClassLoader().loadClass("org.springframework.jdbc.core.JdbcTemplate");
return true;
} catch (ClassNotFoundException e) {
return false;
}
}
//只有当MyCondition的条件成立时,才会创建MyService这个Bean,否则这个Bean的声明就会被忽略掉
@Conditional(MyCondition.class)
public static class MyService {
}
}
Spring Boot本身定义了很多有趣的条件,并把它们运用到了配置类上,这些配置类构成了Spring Boot的自动配置。
- @ConditionalOnBean //配置了某个特定的Bean
- @ConditionalOnMissingBean //没有配置特定的Bean
- @ConditionalOnClass //Classpath里有指定的类
- @ConditionalOnMissingClass //Classpath里缺少指定的类
- @ConditionalOnExpression //给定的Spring Expression Language(SpEL)表达式计算结果为true
- @ConditionalOnJava //java版本匹配特性值或者一个范围
- @ConditionalOnJndi //参数中给定的jndi位置必须存在一个,如果没有给参数则要有JNDI InitialContext
- @ConditionalOnProperty //给定的配置属性要有一个明确的值
- @ConditionalOnResource // Classpath下要有指定的资源
- @ConditionalOnWebApplication // 这是一个web应用程序
- @ConditionalOnNotWebApplication // 这不是一个web应用程序
看个实例
@Configuration
@ConditionalOnClass({ DataSource.class, EmbeddedDatabaseType.class }) //满足这个条件的情况下,配置才会生效
@EnableConfigurationProperties(DataSourceProperties.class)
@Import({ DataSourcePoolMetadataProvidersConfiguration.class,
DataSourceInitializationConfiguration.class }) // 从其他配置类导入额外的配置
public class DataSourceAutoConfiguration {
还有其他例如
- Classpath下有Spring MVC(归功于Web起步依赖),就会配置Spring的DispatherServlet并启用Spring MVC
- Classpath下有Tomcat(通过Web起步依赖的传递引用)就会启动一个嵌入式的Tomcat容器,监听8080端口。
覆盖默认配置
Spring Boot自动配置的Configuration类里@ ConditionalOnMissingBean,只有没有显式去配置才会走自动,但是如果就想覆盖一个简单的参数放弃自动配置显然不合理,如果想微调的话,可以在环境变量、Java系统属性、JNDI、命令行参数、属性文件中指定。
Spring Boot能从多种属性源获取属性,优先级从高到底如下:
- 命令行参数
- java:comp/env里的JNDI属性
- JVM系统参数 , 例如java -jar app.jar --name="dgsa"
- 操作系统环境变量
- 随机生成的带random.*前缀的属性,在设置其他属性时可以引用它们,比如${random.long}
- 应用程序以外的application.properties或者application.yml文件
- 打包在应用程序内的application.properties或者application.yml文件
- 通过@PropertySource标注的属性源
- 默认属性
application.properties或者application.yml可以放在四个位置,优先级如下:
- 外置,在相对于应用程序运行目录的/config子目录里
- 外置,在应用程序运行的目录里
- 内置,在config包内
- 内置,在Classpath根目录
在同一优先级位置,application.yml里的属性会覆盖application.properties里的属性
有300多个属性可以微调Spring Boot应用程序里的Bean。
Spring Boot默认使用logback来记录日志,并且INFO级别输出到控制台。
如果你想用其他日志实现(例如log4j、log4j2)替换logback,引入对应日志实现的起步依赖,同时排除掉logback。
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter</artifactId>
<exclusions>
<exclusion>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-logging</artifactId>
</exclusion>
</exclusions>
</dependency>
也可以完全掌控日志配置,可以在Classpath的根目录src/main/resources里创建logback.xml文件。
如果默认配置能满足,想修改个别配置可以在yml创建logging开头的属性,
logging:
config:
classpath:logging-config.xml // logback配置文件名称
path:/var/logs/ //自定义日志路径
file:x.log // log文件名
level:
root:WARN // root是日志名称
org.springframework.security:DEBUG // 设置此包的日志级别
配置数据源
spring:
datasource:
url: **
username:*
password:**
spring Boot也提供了默认的错误页,不再赘述