Spring Boot 笔记 (1) - Maven、基本配置、Profile的使用
一. Spring Boot 简介
开箱即用的一站式 Java EE 解决方案
Spring 技术栈的大整合
核心问题
暂时无法回答
Spring Boot 和 SOA 有什么区别?
Spring Boot 与 Spring Cloud 的区别与关系?
SpringBoot 做了什么应对微服务的趋势?
其他问题
微服务是什么?
SOA 是什么?
Serveless 又是什么?
Spring Cloud 是什么?
二. Maven
更多 Maven相关内容, 可参考 Maven 笔记 - 难以想象的晴朗
完整 POM 文件
<?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.imzhizi.work</groupId>
<artifactId>xingren</artifactId>
<version>1.2.0-SNAPSHOT</version>
<!--Spring Boot 的父依赖, 统一控制版本-->
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>2.0.7.RELEASE</version>
</parent>
<!--构建打包时的编码和 Java 版本-->
<properties>
<project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
<project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
<java.version>1.8</java.version>
</properties>
<!--通过该插件, 项目将打包为 xingren-1.2-SNAPSHOT.jar -->
<!--项目运行命令为 mvn spring-boot:run -->
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
</plugins>
</build>
<dependencies>
<!--一些 Spring Boot 依赖-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter</artifactId>
</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>
<!--数据库-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-jpa</artifactId>
</dependency>
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
</dependency>
<!--一些推荐的工具库-->
<!-- getter、setter 省略工具 -->
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<optional>true</optional>
</dependency>
<!--JSON 处理工具-->
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>fastjson</artifactId>
<version>1.2.47</version>
</dependency>
<!--DTO 处理工具-->
<dependency>
<groupId>org.modelmapper</groupId>
<artifactId>modelmapper</artifactId>
<version>2.3.0</version>
</dependency>
<!-- JWT 鉴权工具类-->
<dependency>
<groupId>io.jsonwebtoken</groupId>
<artifactId>jjwt</artifactId>
<version>0.9.1</version>
</dependency>
</dependencies>
</project>
其他配置
<groupId>edu.北京邮电大学</groupId>
<artifactId>coope</artifactId>
<version>1.0-SNAPSHOT</version>
<!--表明打包为 war 包-->
<packaging>war</packaging>
<build>
<!--决定了最终生成的包的名称, 默认为 项目名+版本-->
<finalName>coope</finalName>
</build>
三. 核心注解
SpringBootApplication
@SpringBootApplication
||
@SpringBootConfiguration // springboot 配置类, 代表以前的 xml
@EnableAutoConfiguration // 开启自动配置功能, 配置上面的 springboot
# 其中
@EnableAutoConfiguration
||
@AutoConfigurationPackage // 自动配置包
@Import({EnableAutoConfigurationImportSelector.class}) // 导入自动配置类
public @interface EnableAutoConfiguration // 里面放满了各种AutoConfiguration 类名
# 其中
@AutoConfigurationPackage
||
@Import({Registrar.class})
// 自动配置包使用了 Spring 的底层注解 import, 给容器中导入Registrar.class
// 将 SpringBootApplication 所标注的类全部扫描到 Spring 容器中
// 哪里标注了哪些类需要加载呢? 或许是下面的选择器(Selector)
@Import({EnableAutoConfigurationImportSelector.class})
||
EnableAutoConfigurationImportSelector
||继承
AutoConfigurationImportSelector
// 会把该场景需要的所有自动配置类扫描进来 annotationMetadata
// 此类含有 getAutoConfigurationImportListeners 方法
// 此方法使用SpringFactoriesLoader.loadFactories() 加载配置类
// 工厂会完成自动配置工作(把这些自动配置类配置的东西标记为Bean, 然后加载进去)
Controller
一些常见的注解, 没什么新奇的
@Controller
@RestController
@GetMapping
@PostMapping
@PutMapping
@DeleteMapping
@RequestMapping
@PathVarible
@RequestParam
@RequestBody // 用于实体 JSON 类型数据的实例化
@ModelAttributes // 用于非 JSON 类型的数据, 尤其是多媒体文件
@Service
@Repository
@Entity
@AutoWired
@Bean
四. SpringBoot 配置
配置文件, 两个均可使用, 用于 修改默认的配置
- applicaiton.properties
- application.yml (递归命名, 以数据为中心, 更适合做配置文件)
通过此注解
- 可以把配置中的每一个值配置到组件中
- 通过 presfix 和配置中的前缀相匹配
- 既然是组件中, 所以配置类必须声明为组件
配置文件加载顺序
SpringApplication 将从以下位置的 application.properties 文件加载属性,并将它们添加到Spring Environment, 按照以下顺序优先级从高到低, 高优先级会覆盖低优先级的配置
- ./config 项目根目录的config目录
- 项目根目录 ./
- A 类路径 classpath/config 包
- A 类路径根 classpath/
- 为生成的 jar 启动时指定参数
--spring.config.location="config path"
来追加配置, 与之前的默认配置形成互补
外部配置文件加载
-
命令行参数
所有的配置都可以在命令行上进行指定
多个配置用空格分开; --配置项=值
java -jar spring-boot-02-config-02-0.0.1-SNAPSHOT.jar --server.port=8087 --server.context-path=/abc
-
来自 java:comp/env 的JNDI属性
-
Java系统属性(System.getProperties())
-
操作系统环境变量
-
RandomValuePropertySource配置的random.*属性值
-
jar包外部的application-{profile}.properties或application.yml(带spring.profile)配置文件
-
jar包内部的application-{profile}.properties或application.yml(带spring.profile)配置文件
再来加载不带profile
-
jar包外部的application.properties或application.yml(不带spring.profile)配置文件
-
jar包内部的application.properties或application.yml(不带spring.profile)配置文件
-
@Configuration注解类上的@PropertySource
-
通过SpringApplication.setDefaultProperties指定的默认属性
配置文件 YAML 用法
为什么不用 applicaiton.properties, 主要因为 applicaiton.properties 需要写很多遍 prefix
# yaml 写法
# 简单值
k: v # 字符串默认不加引号
k: 'hello \n world' # 原形输出
k: "hello \n world" # 转义输出
# 对象, map 写法
k:
sk: sv # 换行缩进写出
sub-name1:
subName2: # 效果是一样的
k: {sk1: sv1, sk2: sv2}
# 数组写法
k: {v1,v2} # 大括号写法
k: # 换行缩进写法
- v1
- v2
配置文件占位符
## 随机数
my.secret=${random.value}
my.number=${random.int}
my.bignumber=${random.long}
my.uuid=${random.uuid}
my.number.less.than.ten=${random.int(10)}
my.number.in.range=${random.int[1024,65536]}
## 占位符
my.secret=${random.value}
my.number=${my.secret:my-secret}
# 读取配置文件中的其他值, 通过:指定该值不存在时的默认值
配置信息读取
配置注入 @ConfigurationProperties
maven 需要导入
spring-boot-configuration-processor
@Component // 声明为组件, 会被自动加载到 Spring 容器中
@ConfigurationProperties(prefix="config-test") // 从文件中读取配置的类
public class ConfigurationTest {
private String name;
}
值注入注解 @Value
// 通过 ${} 从配置文件, 环境中加载值
// 通过 #{} 使用 SpEL 表达式来计算值
public class ConfigurationTest {
@Value("${config-test.name}")
private String name;
}
配置类测试
使用 SpringBoot 自带的测试框架来替代 JUnit
@RunWith(SpringRunner.class)
@SpringBootTest
public class XxxTest{
@AutoWired
Xxx xxx;
@Test
public void contextLoads(){
// 使用 xxx 进行测试
}
}
@ConfigurationProperties VS @Value
功能 | @ConfigurationProperties | @Value |
---|---|---|
松散绑定 | 支持 | 支持 |
SpEL | 不支持 | 支持 |
JSR303 数据校验 | 支持 | 不支持 |
元数据 | 支持 | 不支持 |
批量绑定 | 支持 | 不支持 |
复杂类型封装 | 支持 | 不支持 |
松散绑定是指 lastName == last-name == last_name == LAST-NAME
所以 @Value 一般是用于取到某项值
而 @ConfigurationProperties 用于真正的配置类
校验注解 @Validated 用于数据校验, 直接加在属性值上, 比如说 @Email
其他来源的配置文件@PropertySource @ImportResource
默认加载的配置都是项目的全局配置文件, 假如想把一些与全局无关的配置抽取出来, 读取此新建配置文件时就需要使用 PropertySource 注解来指定.
@PropertySource(value = {"classpath:xxxx.properties"})
当使用了 Spring 早期的 xml 配置文件, 希望把该 配置文件加再进来, 那么就要用 @ImportResource 导入进来
@ImportResource(locations = {"classpath:beans.xml"})
配置类 @Configuration - 消灭 xml !!!
@Configuration
public class MyAppConfig{
@Bean // 代替 xml 来进行配置, 添加组件, 其实@Component 可以直接注册到容器中
public HelloService helloService(){
// 初始化 Bean 并返回
return helloService;
}
}
五. Profile 实现多配置
多 properties 配置文件方式
# application-dev.properties 文件
server.port=8080
# application-prod.properties 文件
server.port=80
YAML 多文档块方式
通过三个 -
进行文档分块
# application.yml 文件
server:
profiles:
active: name(dev 或 prod)
---
server:
port: 8080
profiles: dev
---
server:
port: 80
profiles: prod
激活不同 Profile 的方法
# 配置文件中声明
spring.profiles.active = 要激活的配置文件
spring:
profiles:
active: 要激活 profile
# 参数控制
# 在运行 jar 时传参激活
--spring.profiles.active=dev
# 指定 Java 虚拟机参数激活
-Dspring.profiles.active=dev