目标:
- @ConfigurationProperties
- 宽松绑定/松散绑定
- 常用计量单位绑定
- 数据校验
1.@ConfigurationProperties
@ConfigurationProperties
在学习yml的时候我们了解到它是可以给对象进行属性注入的,有那么几个前提,yml中有配置,项目中有对应的实体类,用于封装数据且属性名一一对应
我在我的com.example包下创建了一个config的包包中写了这样一个类:
import org.springframework.stereotype.Component;
import lombok.Data;
@Component //因为这个类要被Spring管理,所以加上注解@Component
@Data
public class ServletConfig {
private String ipAddress;
private int port;
private long timeout;
}
yml中
servers:
ipAddress: 192.168.200.140
port: 2345
timeout: -1
我们想为上面的类赋上我们的配置文件的值就需要加上@ConfigurationProperties(prefix = "servers")
@Component //因为这个类要被Spring管理,所以加上注解@Component
@Data
@ConfigurationProperties(prefix = "servers")
public class ServletConfig {
private String ipAddress;
private int port;
private long timeout;
}
验证: 这里选择在引导类中进行测试
import com.example.config.ServletConfig;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.ConfigurableApplicationContext;
@SpringBootApplication
public class ConfigurationLearnApplication {
public static void main(String[] args) {
//拿到容器对象
ConfigurableApplicationContext ctx = SpringApplication.run(ConfigurationLearnApplication.class, args);
ServletConfig bean = ctx.getBean(ServletConfig.class);//拿到这个对象
System.out.println(bean);//打印这个对象
}
}
但是又有一个问题:这个bean是我们自定义的bean,这个bean如果不是自定义的呢?如果是第三方的bean呢?如果这是一个数据源的bean,需要使用第三方bean加载的时候,该怎么做呢?这就是我们接下来要解决的。
假如我们这里引入一个第三方bean:
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>druid</artifactId>
<version>1.2.21</version>
</dependency>
然后我们去定义这个第三方bean,可以再创建一个配置类,但是这里我们就直接使用我们的引导类做演示即可
@SpringBootApplication
public class ConfigurationLearnApplication {
@Bean
public DruidDataSource dataSource() {
DruidDataSource dataSource = new DruidDataSource();
return dataSource;
}
public static void main(String[] args) {
//拿到容器对象
ConfigurableApplicationContext ctx = SpringApplication.run(ConfigurationLearnApplication.class, args);
ServletConfig bean = ctx.getBean(ServletConfig.class);//拿到这个对象
System.out.println(bean);//打印这个对象
DruidDataSource ds = ctx.getBean(DruidDataSource.class);//拿到第三方bean对象
System.out.println(ds);//打印第三方bean对象
}
}
显示的是这个数据源初始化好以后的值,但是只有在druid连接到数据库的时候才会真正的初始化,这个属于是懒加载。
我们手动设置一下
@Bean
public DruidDataSource dataSource() {
DruidDataSource dataSource = new DruidDataSource();
dataSource.setDriverClassName("com.mysql.cj.jdbc.Driver");
return dataSource;
}
在main方法中打印即可验证
System.out.println(ds.getDriverClassName());
你写什么都可以,因为这里还不是真正的连接.这里是我们自己设置的,我们也可以注入进来通过yml,注入到第三方对象上
datasource:
driverClassName: com.mysql.jdbc.Driver456
然后我们到引导类中修改一下:
@SpringBootApplication
public class ConfigurationLearnApplication {
@Bean
@ConfigurationProperties(prefix="datasource")
public DruidDataSource dataSource() {
DruidDataSource dataSource = new DruidDataSource();
// dataSource.setDriverClassName("com.mysql.cj.jdbc.Driver");
return dataSource;
}
public static void main(String[] args) {
//拿到容器对象
ConfigurableApplicationContext ctx = SpringApplication.run(ConfigurationLearnApplication.class, args);
ServletConfig bean = ctx.getBean(ServletConfig.class);//拿到这个对象
System.out.println(bean);//打印这个对象
DruidDataSource ds = ctx.getBean(DruidDataSource.class);//拿到第三方bean对象
System.out.println(ds);//打印第三方bean对象
System.out.println(ds.getDriverClassName());
}
}
@EnableConfigurationProperties
与 @ConfigurationProperties
有什么关系吗?
我们会在SpringBoot中看到大量前面有Enable的东西,我们可以理解为1一种开关,启用那些配置类
在引导类上我们加上这行代码:
@EnableConfigurationProperties(ServletConfig.class) // 启用配置类 如果有多个可以用 {} 包裹起来
结果运行之后发现报错啦!
Exception in thread "main" org.springframework.beans.factory.NoUniqueBeanDefinitionException: No qualifying bean of type 'com.example.config.ServletConfig' available: expected single matching bean but found 2: servletConfig,servers-com.example.config.ServletConfig
matching bean but found 2: 为什么会有两个bean? 在我们前面使用SevletConfig这个类的时候我们使用了@Component
注解,我们把它注释掉,然后@EnableConfigurationProperties
是在告诉spring容器,我们现在有一组信息要去加载配置,如果这个东西它不被spring容器管理那它如何加载这些配置呢?
因此,这里自动把Servlet.class注入到了Spring容器中.
小结:@ConfigurationPropperties
可以为第三方1bean绑定属性
2.宽松绑定/松散绑定
但是如果我们先在配置文件yml中将datasource改成了dataSource
然后再在@ConfigurationProperties(prefix = “dataS
ource”)(改完后) ,
会发现报错了绑定不上
@ConfigurationProperties对于名称上的绑定其实是非常灵活的,Spring为了满足各种开发者的习惯,它提供了多种名称规范.
servers:
ipAddress: 192.168.200.140
port: 2345
timeout: -1
配置类:
@Data
@ConfigurationProperties(prefix = "servers")
public class ServletConfig {
private String ipAddress;
private int port;
private long timeout;
}
ipAddress一一对应是能用的.
但其实配置文件中的ipAdress其实是可以改动的.比如:ipaddress,ip_address,ip-address,IPADDRESS,IP_ADDRESS,IP_ADD_R_E_SS,IP_ADD_R_S-S这些都可以,但是主流格式是:ip-address 它有个有趣的名字叫烤肉串模式
注:宽松绑定不支持注解@Value引用单个属性的方式
小结:
- @ConfigurationProperties绑定属性支持属性名宽松绑定
- @Value注解不支持松散绑定
- 绑定前缀命名命名规则
3. 常用计量单位绑定
假如我们在yml中有这样的配置:
server:
timeout: 30000000000
这样不太直观,到底多少位,而且单位是什么?
我们下面来解决这个问题:
JDK8一系列与单位有关的数据类型,其中有一个专门来描述数据范围的Duration
我们在yml 中加入这样一项: serverTimeOut: 3
然后到配置类中: private Duration serverTimeOut;
这里3 代表什么呢? => 3毫秒
我们可以定义单位:
@DurationUnit(ChronoUnit.SECONDS)
private Duration serveTimeout;
也可以配置存储容量 dataSize
@DataSizeUnit(DataUnit.MEGABYTES)//
private DataSize dataSize;
yml:
dataSize: 1024
但是也太难阅读啦~
如果你把dataSize: 10MB
然后把@DataSizeUnit去掉
10485760 / 1024 =10240
10240 / 1024 = 10
这样看起来更直观
4.数据校验
如果我们在配置文件中配置的时候本来想配置端口123的一不小心写成了a,那这样就会报错
那我们就需要做格式校验validation
数据校验
- 开启数据校验有助于系统1安全性,J233规范中JSR303规范定义了一组有关数据校验相关的API
我们只需要使用这个API即可,引入这个坐标
<dependency>
<groupId>javax.validation</groupId>
<artifactId>validation-api</artifactId>
</dependency>
那这套东西相当于什么呢?
我们用java链接数据库用到了JDBC技术,那问题来了,JDBC是一个接口还是实现类呢?是一个接口。我们用的是Mysql的驱动,这个是实现类,JDBC是一个规范,而mysql驱动是一个数据库的实现类,这里如此。validation是一组接口,还需要导入对应的实现依赖
@Max(value = 8888, message = "端口号不能超过8888")
@Min(value = 80, message = "端口号不能小于80")
private int port;
但是很遗憾报错了,因为接口给我们了,我们没有实现它。就像Servelt是Tomcat实现的,JDBC 是mysql驱动实现的,现在要用校验框架,那实现却没有,因此报错。
上面图中有一个such as Hibernate Valiator
这个Hibernate
校验器是个什么东西?
<dependency>
<groupId>org.hibernate.validator</groupId>
<artifactId>hibernate-validator</artifactId>
</dependency>
这里我们就是使用hibernate
框架提供的校验器做实现类,这样运行后就可以正常运行了。如果我们的配置不符合规定会报错误。
除此之外,hibernate也给我们提供了一些东西
这里提供的方法更多
总结:
本章最后一节我们讲一个问题:配置一个数据库的信息,但是一直连不上,报的错误是密码错误,但是用navicat却能够正常连接,这个问题不是数据库的问题,而是springboot的问题,下面我们来模拟这个问题:
datasource:
driverClassName: com.mysql.jdbc.Driver7122
password: 0127
测试类:
@SpringBootTest
class ConfigurationLearnApplicationTests {
@Value("${servers.ipAddress}")
private String msg;
@Value("${datasource.password}")
private String password;
@Test
void contextLoads() {
System.out.println(msg);
System.out.println(password);
}
}
忽略上面那个,忘记注释了。哈哈哈哈
为什么是87呢?如果修改一下配置文件
datasource:
driverClassName: com.mysql.jdbc.Driver7122
password: "0127"
那为什么?
八进制的格式:0(0-7)
十六进制0x(0-9,a-f)
所以懂了叭~这是格式转换问题 !
小结:
注意yml文件中对于数字的定义支持进制书写格式,如需要使用字符串请使用引号明确标注