zoukankan      html  css  js  c++  java
  • SpringBoot【二】 SpringBoot 配置

    SpringBoot 配置

    配置文件

    SpringBoot 使用一个全局的配置文件,配置文件名称是固定的,作用是修改 SpringBoot 自动配置的默认值(底层自动配置的值),有两种方式可以使用:

    • application.properties

      • 语法结构 :key=value
    • application.yaml【推荐使用】

      • 语法结构 :key:空格value

    比如可以在配置文件中修改 Tomcat 默认启动的端口号

    server.port=8081
    

    yaml

    YAML是 "YAML Ain't a Markup Language" (YAML不是一种标记语言)的递归缩写。在开发的这种语言时,YAML 的意思其实是:"Yet Another Markup Language"(仍是一种标记语言)

    这种语言以数据作为中心,而不是以标记语言为重点!

    1、yaml 和 xml

    以前的配置文件,大多数都是使用xml来配置

    • 传统 xml 配置:

      <server>    
          <port>8081<port>
      </server>
      
    • yaml 配置:

      server:  
      	prot: 8080
      

    2、基础语法

    语法要求严格

    1. 空格不能省略;

    2. 以缩进来控制层级关系,只要是左边对齐的一列数据都是同一个层级的;

    3. 属性和值的大小写都是十分敏感的。

    • 字面量:普通的值 [ 数字,布尔值,字符串 ]

      字面量直接写在后面就可以,字符串默认不用加上双引号或者单引号

      k: v
      

      注意:

      • “ ” 双引号,不会转义字符串里面的特殊字符,特殊字符会作为本身想表示的意思

        比如:name: "zhang san" 输出 :zhang 换行 san

      • '' 单引号,会转义特殊字符,特殊字符最终会变成和普通字符一样输出

        比如 :name: ‘zhang san’ 输出 :zhang san

    • 对象、Map(键值对)

      #对象、Map格式
      k:     
      	v1:    
      	v2:
      

      在下一行来写对象的属性和值的关系,注意缩进

      student:    
      	name: zhangsan   
      	age: 3
      	
      # 行内写法
      student: {name: zhangsan,age: 3}
      
    • 数组( List、set )

      用 - 值表示数组中的一个元素,比如:

      pets: 
      	- cat 
      	- dog 
      	- pig
      	
      # 行内写法
      pets: [cat,dog,pig]
      

    注入配置文件

    1、yaml 配置文件注入

    yaml 文件强大的地方在于,它可以给我们的实体类直接注入匹配值。

    1. 在 springboot 项目的 resources 目录下新建一个文件 application.yml

    2. 编写一个实体类 Dog

    3. 原来 Spring 中给 bean 注入属性值,通过 @Value

      @Component //注册bean到容器中
      public class Dog {
          @Value("旺财")
          private String name;
          @Value("3")
          private int age;
          ...//有参无参构造、get、set方法、toString()方法  
      }
      
    4. SpringBoot 的测试类下注入 Dog 对象并输出

      @SpringBootTest
      class Springboot02ConfigApplicationTests {
      
      	@Autowired // 自动注入
      	private Dog dog;
      
         @Test
         void contextLoads() {
            System.out.println(dog); // 输出结果:Dog{name=旺财, age=3}
         }
      }
      
    5. 编写一个复杂一点的实体类:Person 类

      @Component //注册 bean 到容器中
      public class Person {
          
          private String name;
          private int age;
          private boolean happy;
          private Date birth;
          private Map<String,Object> maps;
          private List<Object> lists;
          private Dog dog;
      	... //有参无参构造、get、set方法、toString()方法  
      }
      
    6. 使用 yaml 配置的方式进行注入,编写 application.yml

      person:
        name: zhangsan
        age: 3
        happy: true
        birth: 2020/02/02
        maps: {k1: v1,k2: v2}
        lists:
          - code
          - music
          - dance
        dog:
          name: wangcai
          age: 3
      
    7. 注入到 Person 类中

      @Component
      @ConfigurationProperties(prefix = "person")
      public class Person {
      	... // 不变
      }
      

      @ConfigurationProperties 作用:

      将配置文件中配置的每一个属性的值,映射到这个组件中,告诉 SpringBoot 将本类中的所有属性和配置文件中相关的配置进行绑定,其中参数 prefix = “person” 表示将与配置文件中的 person 下面的所有属性一一对应。

    8. 标注 @ConfigurationProperties 注解后,IDEA 提示报红(但是不影响程序运行),SpringBoot 配置注解处理器没有找到,查看文档,找到一个依赖

      <!-- 导入配置文件处理器,配置文件进行绑定就会有提示,需要重启 -->
      <dependency>
         <groupId>org.springframework.boot</groupId>
         <artifactId>spring-boot-configuration-processor</artifactId>
         <optional>true</optional>
      </dependency>
      
    9. 测试类中测试

      @SpringBootTest
      class Springboot02ConfigApplicationTests {
      
         @Autowired
         private Person person;
      
         @Test
         void contextLoads() {
            System.out.println(person);
         }
      
    10. 输出 Person 对象,属性值和 yaml 注入的相同,说明所有值全部注入成功

      Person{name='zhangsan', age=3, happy=true,birth=Sun Feb 02 00:00:00 GMT+08:00 2020,maps={k1=v1, k2=v2}, lists=[code, music, dance],dog=Dog{name='wangcai', age=3}}
      

    注意:当配置文件的 key 值和属性的值设置的不一样时,结果输出为 null,注入失败。

    配置文件占位符:配置文件可以编写占位符生成随机数

    person:
     name: zhangsan${random.uuid} # 随机uuid
     age: ${random.int}  # 随机int
     happy: true
     birth: 2020/02/02
     maps: {k1: v1,k2: v2}
     lists:
       - code
       - music
       - dance
     dog:
       name: ${person.hello:other}_wangcai
       age: 3
    

    2、properties 配置文件注入(加载指定的配置文件)

    properties 配置文件在写中文的时候,会有乱码,需要去 IDEA 中(settings --> FileEncodings)设置编码格式为UTF-8。

    @PropertySource :加载指定的配置文件

    @configurationProperties:默认从全局配置文件中获取值

    1. 在 resources 目录下新建一个 person.properties 文件

      name=lisi
      
    2. 然后在 Person 类中指定加载 person.properties 文件

      @Component
      // 加载指定的配置文件
      @PropertySource(value = "classpath:application.properties")
      public class Person {	//直接使用@value  
          
          // @Value("男")  // 字面量 
          // @Value("#{9*2}")  // #{SPEL} Spring表达式
          @Value("${name}") // spring EL表达式取出配置文件的值
          private String name;
          private int age;
          ...
      }
      
    3. 测试输出,Person 对象中只有 name 属性被赋值为 lisi,其他都为 null,说明配置文件绑定成功。

      Person{name='lisi', age=null, happy=null, birth=null, maps=null, lists=null, dog=null}
      

    注意:@Value 使用起来并不友好,因为我们需要为每个属性单独注解赋值,比较麻烦

    3、两种方式对比

    1. @ConfigurationProperties 只需要写一次即可, @Value 则需要每个字段都添加

    2. 松散绑定:比如 yml 中写的 last-name,和实体类中的 lastName 一样, - 后面跟着的字母默认是大写的。

    3. JSR303 数据校验 , 可以在字段上增加一层过滤器验证,可以保证数据的合法性

    4. 复杂类型封装,yml 中可以封装对象 , 使用 value 就不支持

    结论:

    • 配置 yml 和配置 properties 都可以获取到值 , 强烈推荐 yml;

    • 如果在某个业务中,只需要获取配置文件中的某个值,可以使用一下 @value;

    • 如果专门编写了一个 JavaBean 来和配置文件进行一一映射,使用 yaml 和 @configurationProperties。

    JSR303 数据校验

    Springboot 中可以用 @validated 来校验数据,如果数据异常则会统一抛出异常,方便异常中心统一处理。使用数据校验,可以保证数据的正确性。

    在 Person 类中加入注解让 name 只能支持 Email 格式;

    @Component
    @ConfigurationProperties(prefix = "person")
    @Validated // 数据校验
    public class Person {
    
        @Email(message = "邮箱格式错误") //name必须是邮箱格式
        private String name;
    	...
    }
    

    运行结果:default message [不是一个合法的电子邮件地址];

    常见参数

    @NotNull(message="名字不能为空")
    private String userName;
    @Max(value=120,message="年龄最大不能查过120")
    private int age;
    @Email(message="邮箱格式错误")
    private String email;
    
    空检查
    @Null       验证对象是否为null
    @NotNull    验证对象是否不为null, 无法查检长度为0的字符串
    @NotBlank   检查约束字符串是不是Null还有被Trim的长度是否大于0,只对字符串,且会去掉前后空格.
    @NotEmpty   检查约束元素是否为NULL或者是EMPTY.    
    
    Booelan检查
    @AssertTrue     验证 Boolean 对象是否为 true  
    @AssertFalse    验证 Boolean 对象是否为 false      
    
    长度检查
    @Size(min=, max=) 验证对象(Array,Collection,Map,String)长度是否在给定的范围之内  @Length(min=, max=) string is between min and max included.
    
    日期检查
    @Past       验证 Date 和 Calendar 对象是否在当前时间之前  
    @Future     验证 Date 和 Calendar 对象是否在当前时间之后  
    @Pattern    验证 String 对象是否符合正则表达式的规则
    .......等等除此以外,我们还可以自定义一些数据校验规则
    

    多环境切换

    profile 是 Spring 对不同环境提供不同配置功能的支持,可以通过激活不同的环境版本,实现快速切换环境。

    1、多配置文件

    在主配置文件编写的时候,文件名可以是 application-{profile}.properties/yml , 用来指定多个环境版本

    例如:

    • application-test.properties 代表测试环境配置
    • application-dev.properties 代表开发环境配置

    但是 Springboot 并不会直接启动这些配置文件,它默认使用 application.properties主配置文件,需要通过一个配置来选择需要激活的环境:

    #比如在配置文件中指定使用dev环境,我们可以通过设置不同的端口号进行测试;
    #我们启动SpringBoot,就可以看到已经切换到dev下的配置了;
    spring.profiles.active=dev
    

    2、yaml 的多文档块

    和 properties 配置文件中一样,但是使用 yml 去实现不需要创建多个配置文件,更加方便。

    server:  
    	port: 8081
    	
    #选择要激活那个环境块
    spring:  
    	profiles:    
    		active: prod
    		
    ---
    server:  
    	port: 8083
    spring:  
    	profiles: dev #配置环境的名称
    
    ---
    server:  
    	port: 8084
    spring:  
    	profiles: prod  #配置环境的名称
    

    注意:如果 yml 和 properties 同时都配置了端口,并且没有激活其他环境,默认会使用 properties 配置文件。

    配置文件加载位置

    官方文档说明,外部配置文件位置有:

    1. file:./config/

    2. file:./

    3. classpath:./config/

    4. classpath:./

    SpringBoot 启动会扫描以下位置的 application.properties 或者 application.yml 文件作为 SpringBoot 的默认配置文件,优先级由高到底,高优先级的配置会覆盖低优先级的配置:

    优先级1:项目路径(file)下的config文件夹配置文件
    优先级2:项目路径下配置文件
    优先级3:资源路径(classpath)下的config文件夹配置文件
    优先级4:资源路径下配置文件
    

    SpringBoot 会从这四个位置全部加载主配置文件,互补配置。

    指定位置加载配置文件:

    可以通过 spring.config.location 来改变默认的配置文件位置,项目打包好以后,可以使用命令行参数的形式,启动项目的时候来指定配置文件的新位置,这种情况,一般是后期运维做的多,相同配置,外部指定的配置文件优先级最高

    java -jar spring-boot-config.jar --spring.config.location=F:/application.properties
    

    自动配置原理

    关注两种文件:

    • xxxAutoConfiguration:自动配置类,给容器中添加组件

    • xxxProperties :默认属性类,封装配置文件中相关属性,和配置文件绑定,这样就可以使用自定义的配置

    原理分析:

    HttpEncodingAutoConfiguration(Http编码自动配置)和 ServerProperties 为例:

    // 表示这是一个配置类,和以前编写的配置文件一样,也可以给容器中添加组件
    @Configuration(proxyBeanMethods = false)
    
    // 自动配置属性:ServerProperties
    	//进入这个ServerProperties查看,将配置文件中对应的值和ServerProperties绑定起来;
        //并把ServerProperties加入到ioc容器中
    @EnableConfigurationProperties(ServerProperties.class)
    
    //Spring底层@Conditional注解
        //根据不同的条件判断,如果满足指定的条件,整个配置类里面的配置就会生效;
        //这里的意思就是判断当前应用是否是web应用,如果是,当前配置类生效
    @ConditionalOnWebApplication(type = ConditionalOnWebApplication.Type.SERVLET)
    
    //判断当前项目有没有这个类CharacterEncodingFilter:SpringMVC中进行乱码解决的过滤器
    @ConditionalOnClass(CharacterEncodingFilter.class)
    
    //判断配置文件中是否存在某个配置:server.servlet.encoding
        //如果不存在,判断也是成立的
        //即使我们配置文件中不配置 server.servlet.encoding=true,也是默认生效的;
    @ConditionalOnProperty(prefix = "server.servlet.encoding", value = "enabled", matchIfMissing = true)
    public class HttpEncodingAutoConfiguration {
    
       //它已经和SpringBoot的配置文件映射了
       private final Encoding properties;
    
       //只有一个有参构造器的情况下,参数的值就会从容器中拿
       public HttpEncodingAutoConfiguration(ServerProperties properties) {
          this.properties = properties.getServlet().getEncoding();
       }
    
       //给容器中添加一个组件,这个组件的某些值需要从properties中获取
       @Bean
       @ConditionalOnMissingBean
       public CharacterEncodingFilter characterEncodingFilter() {
          CharacterEncodingFilter filter = new OrderedCharacterEncodingFilter();
          filter.setEncoding(this.properties.getCharset().name());
    	filter.setForceRequestEncoding(this.properties.shouldForce(Encoding.Type.REQUEST));
    		filter.setForceResponseEncoding(this.properties.shouldForce(Encoding.Type.RESPONSE));
    		
          return filter;
       }
        ...
    }
    
    //从配置文件中获取指定的值和 bean 的属性进行绑定
    @ConfigurationProperties(prefix = "server", ignoreUnknownFields = true)
    public class ServerProperties {
        ...
    }
    

    总结:

    根据当前不同的条件判断,决定这个配置类是否生效;一旦这个配置类生效,这个配置类就会给容器中添加各种组件;这些组件的属性是从对应的 properties 类中获取的,这些类里面的每一个属性又是和配置文件绑定的;

    所有在配置文件中能配置的属性都是在 xxxxProperties 类中封装着,配置文件能配置什么就可以参照某个功能对应的这个属性类。

    精髓:

    1. SpringBoot 启动会加载大量的自动配置类
    2. 看我们需要的功能有没有在 SpringBoot 默认写好的自动配置类当中;
    3. 再来看这个自动配置类中到底配置了哪些组件(只要需要用的组件存在其中,我们就不需要再手动配置)
    4. 给容器中自动配置类添加组件的时候,会从 properties 类中获取某些属性,我们只需要在配置文件中指定这些属性的值即可。

    @Conditional

    自动配置类必须在一定的条件下才能生效,主要原因是每个自动配置类上都添加了 @Conditional 的派生注解。

    Spring 注解版原生的 @Conditional 作用:必须是 @Conditional 指定的条件成立,才给容器中添加组件,配置里面的所有内容才生效。

    问题:如何知道哪些自动配置类生效?

    通过启用 debug=true 属性,来让控制台打印自动配置报告,这样知道哪些自动配置类生效

    #开启springboot的调试类
    debug=true
    

    输出日志中将配置类分为三种:

    • Positive matches:(自动配置类启用的:正匹配)

    • Negative matches:(没有启动,没有匹配成功的自动配置类:负匹配)

    • Unconditional classes: (没有条件的类)

  • 相关阅读:
    前端试题本(Javascript篇)
    前端知识杂烩(Javascript篇)
    前端知识杂烩(HTML[5]?+CSS篇)
    Javascript实现的数组降维——维度不同,怎么谈恋爱
    你不知道的CSS背景—css背景属性全解
    基于Codeigniter框架实现的APNS批量推送—叮咚,查水表
    CSS布局经典—圣杯布局与双飞翼布局
    JavaScript异步编程的主要解决方案—对不起,我和你不在同一个频率上
    CSS实现元素水平垂直居中—喜欢对称美,这病没得治
    JS实现常用排序算法—经典的轮子值得再造
  • 原文地址:https://www.cnblogs.com/Songzw/p/13269448.html
Copyright © 2011-2022 走看看