整个项目中所有重要的数据都是在配置文件中配置的,比如:
数据库的连接信息(包含用户名和密码的设置);
项目的启动端口;
第三方系统的调用秘钥等信息;
用于发现和定位问题的普通日志和异常日志等。
想象一下如果没有配置信息,那么 Spring Boot 项目就不能连接和操作数据库,甚至是不能保存可以用于排查问题的关键日志,所以配置文件的作用是非常重要的
系统的配置文件,比如连接字符串、比如日志的相关设置,是系统定义好的
用户自定义的
Spring Boot 配置文件主要分为以下两种格式:
.properties
.yml
这就好像连锁店里面的统一服装一样,有两种不同的款式,properties 类型的配置文件就属于老款“服饰”,也是创建 Spring Boot 项目时默认的文件格式(主要是由于仓库里还有库存),而 yml 属于新版款式,如果用户了解情况直接指定要新款服饰,那么就直接发给他
理论上讲 properties 可以和 yml 一起存在于一个项目当中,当 properties 和 yml 一起存在一个项目中时,如果配置文件中出现了同样的配置,比如 properties 和 yml 中都配置了“server.port”,那么这个时候会以 properties 中的配置为主,也就是 .properties 配置文件的优先级最高,但加载完 .properties 文件之后,也会加载 .yml 文件的配置信息。
虽然理论上来讲 .properties 可以和 .yml 共存,但实际的业务当中,我们通常会采取一种统一的配置文件格式,这样可以更好的维护(降低故障率),减少开发者难度。这就好像连锁店的服饰一样,不管是老款的服装还是新款的服装,一定要统一了才好看
如果 properies 文件没有提升信息,那么就需要安装spring tools插件了
IDEA 社区版安装 Spring Assistant 插件之后,就可以正常创建 Spring Boot 项目了,并且 yml 的配置文件就有提示了。但默认情况下是不⽀持 properties 格式的日志提示的,这个时候需要安装了 SpringTools 插件才会有相应的提示
properties 配置文件是最早期的配置文件格式,也是创建 Spring Boot 项目默认的配置文件
properties 是以键值的形式配置的,key 和 value 之间是以 “=” 连接的,key=value
,如 application.properties:
# 系统的配置文件
server.port=9090
spring.datasource.url=jdbc:mysql://127.0.0.1:3306/testdb?characterEncoding=utf8mb4&useSSL=true
characterEncoding=utf8
spring.datasource.username=root
spring.datasource.password=123
# 自定义配置项
cctalk.token=xxx
想要查看 Spring Boot 更多系统配置项,访问官网:Common Application Properties
PS:不加空格,可能会被当做 value 的一部分
小技巧:配置文件中使用“#”来添加注释信息
1). 配置当前项目中文乱码的问题 Settings
2). 配置新创建项目的中文乱码问题 New Projects Settings
3). 将原来的 application. properies 删除掉,重新再创建一个 application. properieso
因为之前的配置文件已经是不支持中文编码的文件格式化,设置编码再创建 application.properies 之后,所以后设置的编码不会改变原来的旧的 application.properies 文件,所以需要先将之前的配置文件删除掉重新创建。
经过上面的 3 步之后才能创建一个不是中文乱码的 application. properies 文件,并且以后在创建 application. properies 不会出现中文乱码
如果在项目中,想要主动的读取配置文件中的内容,可以使用 @Value 注解来实现。
@Value 注解使用 “${}” 的格式读取,如下代码所示
@Controller
public class UserController {
@Value("${server.port}") // 读取配置项
private String port;
@ResponseBody
@RequestMapping("/sayhi")
public String sayhi() {
return "hello, word —— port: " + port;
}
}
访问 http://localhost:8080/sayhi,最终执行效果:hello, word —— port: 8080
@Component 在 Spring Boot 启动时候会注入到框架中,注入到框架中时会执行 @PostConstruct 初始化方法,这个时候就能读取到配置信息了
读取多个配置文件:
properties 配置是以 key-value 的形式配置的
从上述配置key看出,properties 配置文件中会有很多的冗余的信息:
spring.datasource.username=root
spring.datasource.password=123
想要解决这个问题,就可以使用 yml 配置文件的格式化了
yml 是 YAML 是缩写,它的全称 Yet Another Markup Language 翻译成中文就是 “另一种标记语言”
yml 是一个写法简单,可读性高,易于理解,用来表达数据序列化的格式。它的语法和其他高级语言类似,
支持更多的数据类型,可以简单表达清单(数组)、散列表,标量等数据形态。它使用空白符号缩进和大量依赖外观的特色,特别适合用来表达或编辑数据结构、各种配置文件等。
yml 最大的优势是可以跨语言,不止是 Java 中可以使用 golang、python 都可以使用 yaml 作为配置文件
yml 是树形结构的配置文件,它的基础语法是“key: value”,注意 key 和 value 之间使用英文冒汗加空格的方式组成的,其中的空格不可省略。
基础语法如下:
其中第一项的配置为正确的,key 也是高亮显示的,而第二项没有空格是错误的使用方式,第二项的key 也没有高亮显示
application.yml:
# 连接数据库
spring:
datasource:
url: jdbc:mysql://127.0.0.1:3306/testdb?characterEncoding=utf8mb4&useSSL=true
name: root
password: 123
yml 和 properties 连接数据库的配置对比:
# 字符串
string.value: Hello
# 布尔值,true或false
boolean.value: true
boolean.value1: false
# 整数
int.value: 10
int.value1: 0b1010_0111_0100_1010_1110 # 二进制
# 浮点数
float.value: 3.14159
float.value1: 314159e-5 # 科学计数法
# Null,~代表null
null.value: ~
yml 读取配置的方式和 properties 相同,使用 @Value 注解即可,实现代码如下:
@Component
public class ReadYml {
@Value("${string.hello}")
private String hello;
@PostConstruct
public void postConstruct() {
System.out.println("Read YML,Hello:" + hello);
}
}
使用代码测试一下,读取值为 null 的配置和根本不存在的配置结果是否一致?
字符串默认不用加上单引号或者双引号,如果加英文的单双引号可以表示特殊的含义
尝试在 application.yml 中配置如下信息:
mystring1: hello\nword
mystring2: 'hello\nword'
mystring3: "hello\nword"
读取程序实现代码如下:
@Controller
public class UserController {
@Value("${mystring1}") // 读取配置项
private String mystring1;
@Value("${mystring2}")
private String mystring2;
@Value("${mystring3}")
private String mystring3;
@ResponseBody
@RequestMapping("/sayhi")
public String sayhi() {
System.out.println("mystring1: " + mystring1);
System.out.println("mystring2: " + mystring2);
System.out.println("mystring3: " + mystring3);
}
}
访问 http://localhost:8080/sayhi,执行结果:
如果在 mystring3 中加入转义符 \
:mystring3: "hello\\nword"
结果:mystring3: hello\nword
从上述结果可以看出:
——application.yml:
# 对象写法 1:
student:
id: 1
name: 张三
age: 20
# 对象写法 2:行内写法(与上面的写法作用一致)
student2: {id: 2,name: 李四,age: 20}
这个时候就不能用 @Value 来读取配置中的对象了,此时要使用另一个注解 @ConfigurationProperties
来读取 常见类 Student,具体实现如下:
@Data
中包含了 getter,setter,toString 等
import lombok.Data;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.stereotype.Component;
@Data
@ConfigurationProperties(prefix = "student") // 读取配置文件中的对象 在项目启动时读取
@Component // 不能省略
public class Student {
private int id;
private String name;
private int age;
}
——调用类的实现如下:
@Controller
public class UserController {
@Autowired
private Student student; // 属性注入
@ResponseBody
@RequestMapping("/sayhi")
public String sayhi() {
return "student: " + student;
}
}
访问,执行效果:student: Student(id=1, name=张三, age=20)
读李四:修改Student 类:@ConfigurationProperties(prefix = “student2”)。
——application.yml:
# 集合写法 1:
dbtypes:
name:
- mysql
- sqlserver
- db2
# 集合写法 2:行内写法(与上面的写法作用一致)
dbtype: {name: [mysql,sqlserver,db2]}
——集合的读取和对象一样,也是使用 @ConfigurationProperties 来读取的,创建类 ReadList 具体实现如下:
@Component
@Data
@ConfigurationProperties("dbtype") // 读取配置文件中的集合
public class ReadList {
private List<String> name;
}
——打印类的实现如下:
@Controller
public class UserController {
@Resource
private ReadList readList;
@ResponseBody
@RequestMapping("/sayhi")
public String sayhi() {
return "readList: " + readList.getName();
}
}
访问,执行效果:readList: [mysql, sqlserver, db2]
properties 是以 key=value 的形式配置的键值类型的配置文件,而 yml 使用的是类似 json 格式的树形配置方式进行配置的,yml 层级之间使用换行缩进的方式配置,key 和 value 之间使用“: ”英文冒号加空格的方式设置,并且空格不可省略。
properties 为早期并且默认的配置文件格式,但其配置存在一定的冗余数据,使用 yml 可以很好的解决数据冗余的问题。
yml 通用性更好,⽀持更多语言,如 Java、Go、Python 等,如果是云服务器开发,可以使用一份配置文件作为 Java 和 Go 的共同配置文件。
yml 虽然可以和 properties 共存,但一个项目中建议使用统一的配置类型文件
Spring Boot 中读取配置文件有以下 5 种方法:
使用@Value 读取配置文件
使用@ConfigurationProperties读取配置文件
使用Environment读取配置文件
使用@PropertySource 读取配置文件
使用@PropertySource 注解可以用来指定读取某个配置文件,比如指定读取application.properties配置文件的配置内容
@PropertySource 注解默认只支持 properties 格式配置文件的读取
@springBootApplication
@PropertySource( "classpath:application.properties", encoding="utf-8")
public class DemoApplication implements InitializingBean {
@Value("${profile.name}")
private string name;
public static void main(String[ ] args) {
SpringApplication.run(DemoApplication.class, args );
}
@Override
public void afterPropertiesSet( ) throws Exception {
System.out.println("Name: " +name ) ;
}