SpringBoot学习之路(五)- yml配置文件、加载顺序、配置原理

版权声明:博客对我来说是记忆的笔记和知识的分享~非常感谢博客大神的帮助,若有无意侵权,请您联系我,谢谢^_^!转载请声明出处: https://blog.csdn.net/qq_36698956/article/details/85994093

前言

主要介绍下SpringBoot中的配置文件、加载顺序及配置原理

一、配置文件

1、配置文件简介

SpringBoot使用一个全局的配置文件,配置文件名是固定的;

  • application.properties
  • application.yml
    配置文件的作用:修改SpringBoot自动配置的默认值;SpringBoot在底层都给我们自动配置好;
    YAML(YAML Ain’t Markup Language)有两种写法:
    YAML A Markup Language:是一个标记语言
    YAML isn’t Markup Language:不是一个标记语言;
    不管他是不是标记语言,但是他跟标记语言都脱不了关系。

标记语言区别:

  • 以前的配置文件;大多都使用的是 xxxx.xml文件;
  • YAML:以数据为中心,比json、xml等更适合做配置文件;

YAML:配置例子

server:
	port: 8081

XML:配置例子

<server>
	<port>8081</port>
</server>

这样我们能看到xml的写法都大量浪费在标签,而yml是以数据为中心的,通过这个简介的例子对比就能看出来。

2、YAML语法

1)、基本语法

k:(空格)v:表示一对键值对 (空格必须有);
空格的缩进来控制层级关系;只要是左对齐的一列数据,都是同一个层级的
属性和值也是大小写敏感

2)、值的写法

字面量:普通的值(数字、字符串、布尔)

k:空格V :字面直接来写

  • 字符串默认不用加上单引号或者双引号;
  • “”:双引号;不会转义字符串里面的特殊字符;特殊字符会作为本身想表示的意识;
    • 举例:name:空格“zhangsan \n lisi” ,输出:zhangsan 换行 lisi
  • ‘’:单引号;会转义特殊字符,特殊字符最终只是一个普通的字符串数据
    • 举例:name:空格’zhangsan \n lisi’ ,输出:zhangsan \n lisi
对象、Map(属性和值)(键值对)

k:空格V : V直接写在下一行来写对象的属性和值的关系;注意缩进

friends:
		lastName: zhangsan
		age: 20

行内写法:

friends: {lastName: zhangsan,age: 18}

大括号{前有个空格,并且值zhangsan前也有空格

数组(List、Set)

用- 值表示数组中的一个元素
举例:

pets:
 - cat
 - dog
 - pig

行内写法:

pets: [cat,dog,pig]

3、配置文件值注入

YMAL配置文件

1)、创建实体类注值

举例:
首先创建两个类Person和Dog

package com.yhhy.springboot.springboot.bean;
import java.util.Date;
import java.util.List;
import java.util.Map;
/**
 * 将配置文件中配置的每一个属性的值,映射到这个组件中
 * @ConfigurationProperties : 告诉SpringBoot将本类中的所有属性和配置文件中相关的配置进行绑定
 * prefix = 这个前缀就是告诉我们要跟yml中那个前缀对应
 * @Component : 只有这个组件是容器中的组件,才能容器提供的@ConfigurationProperties功能;
 */
public class Person {
    private  String  lastName;
    private Integer age;
    private Boolean boss;
    private Date    birth;
    private Map<String,Object> maps;
    private List<Object> lists;
    private Dog dog;
    public String getLastName() {
        return lastName;
    }
    public void setLastName(String lastName) {
        this.lastName = lastName;
    }
    public Integer getAge() {
        return age;
    }
    public void setAge(Integer age) {
        this.age = age;
    }
    public Boolean getBoss() {
        return boss;
    }
    public void setBoss(Boolean boss) {
        this.boss = boss;
    }
    public Date getBirth() {
        return birth;
    }
    public void setBirth(Date birth) {
        this.birth = birth;
    }
    public Map<String, Object> getMaps() {
        return maps;
    }
    public void setMaps(Map<String, Object> maps) {
        this.maps = maps;
    }
    public List<Object> getLists() {
        return lists;
    }
    public void setLists(List<Object> lists) {
        this.lists = lists;
    }
    public Dog getDog() {
        return dog;
    }
    public void setDog(Dog dog) {
        this.dog = dog;
    }
    @Override
    public String toString() {
        return "Person{" +
                "lastName='" + lastName + '\'' +
                ", age=" + age +
                ", boss=" + boss +
                ", birth=" + birth +
                ", maps=" + maps +
                ", lists=" + lists +
                ", dog=" + dog +
                '}';
    }
}

package com.yhhy.springboot.springboot.bean;
public class Dog {
    private String name;
    private Integer age;
    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }
    public Integer getAge() {
        return age;
    }
    public void setAge(Integer age) {
        this.age = age;
    }
    @Override
    public String toString() {
        return "Dog{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}

在这里插入图片描述
可以看到在Person.java上方有警告提示:
在这里插入图片描述
这个不算是报错,只是IDEA的警告提示,他说这个Spring Boot的配置文件的注释处理器没有在类路径下找到。
可以点击这个这个按钮:在这里插入图片描述
可以看到官方文档让我们依赖于这个东西:在这里插入图片描述
这个是配置文件的处理器,是帮我们生成原文件的数据信息,相当于配置了这个之后,在yml中写配置会有提示信息;

在这里插入图片描述

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

这时候再看Person.java的上方此警告就会消失了;

2)、启动测试查看结果

可以使用SpringBoot的单元测试
在这里插入图片描述

properties配置文件

1)、注入值测试结果

首先将application.yml中注入的值先注释掉,在application.properties中进行注入
在这里插入图片描述

2)、properties配置文件在idea中默认utf-8可能会乱码

而以前我们在properties中写的中文都是用ASCII码,所以需要在设置中改一下
在这里插入图片描述
测试:
在这里插入图片描述

3)、@Value注解进行注入值
package com.yhhy.springboot.springboot.bean;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;
import java.util.Date;
import java.util.List;
import java.util.Map;

/**
 * @Component : 只有这个组件是容器中的组件,才能容器提供的@ConfigurationProperties功能;
 */
@Component
//使用@Value进行注值,所以将其注释掉
//@ConfigurationProperties(prefix = "person")
public class Person {
    /**
     * @Value 注值类似于我们之前写的
     * <bean calss="Person">
     *     <property name="lastName" value="字面量"><property/>
     * <bean/>
     * 3中写法:
     *  1:能写字面量
     *  2:${key},他的作用就是从环境变量、配置文件中获取值
     *  3:支持SpEL,spring的表达式语言
     *
     */
    @Value("${person.last-name}")
    private  String  lastName;
    @Value("#{11*2}")
    private Integer age;
    @Value("true")
    private Boolean boss;
    private Date    birth;
    private Map<String,Object> maps;
    private List<Object> lists;
    private Dog dog;
    public String getLastName() {
        return lastName;
    }
    public void setLastName(String lastName) {
        this.lastName = lastName;
    }
    public Integer getAge() {
        return age;
    }
    public void setAge(Integer age) {
        this.age = age;
    }
    public Boolean getBoss() {
        return boss;
    }
    public void setBoss(Boolean boss) {
        this.boss = boss;
    }
    public Date getBirth() {
        return birth;
    }
    public void setBirth(Date birth) {
        this.birth = birth;
    }
    public Map<String, Object> getMaps() {
        return maps;
    }
    public void setMaps(Map<String, Object> maps) {
        this.maps = maps;
    }
    public List<Object> getLists() {
        return lists;
    }
    public void setLists(List<Object> lists) {
        this.lists = lists;
    }
    public Dog getDog() {
        return dog;
    }
    public void setDog(Dog dog) {
        this.dog = dog;
    }
    @Override
    public String toString() {
        return "Person{" +
                "lastName='" + lastName + '\'' +
                ", age=" + age +
                ", boss=" + boss +
                ", birth=" + birth +
                ", maps=" + maps +
                ", lists=" + lists +
                ", dog=" + dog +
                '}';
    }
}

查看application.properties中的写法:
在这里插入图片描述
测试结果:
在这里插入图片描述
可以看出一样可以注入值

4)、@Value获取值和@ConfigurationProperties获取值比较
@ConfigurationProperties @Value
功能 批量注入配置文件中的属性 一个个指定
松散绑定(松散语法) 支持 不支持
SpEL 不支持 支持
JSR303数据校验 支持 不支持
复杂类型封装 支持 不支持
  • 松散绑定(松散语法)
    在Person类中的属性为private String lastName; 而properties的配置文件中声名的key是person.last-name ,也是可以绑定注入值得;
    松散绑定这几种任意写,都是可以注入:在这里插入图片描述

  • SpEL
    @Value("#{11*2}")这种计算,只支持@Value

  • JSR303数据校验
    在注入值得时候,可以对数据进行校验

    • 1、首先给Person类加上@Validated注解
    • 2、例如给属性lastName上加上@Email注解
    • 3、在启动单元测试类报错:
    • 在这里插入图片描述
  • 复杂类型封装
    例如:如果使用@Value给Person类中的private Map<String,Object> maps;注入值时,就会报错
    在这里插入图片描述

PS:
配置文件yml还是properties他们都能获取到值;
如果说,我们只是在某个业务逻辑中需要获取一下配置文件中的某项值,使用@Value;
如果说,我们专门编写了一个javaBean来和配置文件进行映射,我们就直接使用@ConfigurationProperties;

5)、@PropertySource&@ImportResource&@Bean
a、@PropertySource 注解: 加载指定类路径下配置文件

如果把所有东西都配置在全局配置文件中(application.properties),那可能这个配置文件就太大了,我想把一些跟SpringBoot无关的配置想提取出来,这样先写一个person.properties,目前想将application.properties中写的配置都拿到person.properties中

package com.yhhy.springboot.bean;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.PropertySource;
import java.util.Date;
import java.util.List;
import java.util.Map;
/**
 * @ConfigurationProperties(prefix = "person")  默认从全局配置文件中获取
 * @PropertySource(value = {"classpath:person.properties"}) 告诉SpringBoott加载类路径下的person.properties内容
 */
@PropertySource(value = {"classpath:person.properties"})
@Configuration
@ConfigurationProperties(prefix = "person")
public class Person {
    private String lastName;
    private Integer age;
    private Boolean boss;
    private Date birth;
    private Map<String,Object> maps;
    private List<Object> lists;
    private Dog dog;

在这里插入图片描述
测试结果依然可以拿到的。
@PropertySource的参数可以是数组,可以加载多个配置文件

b、@ImportResource 注解:导入Spring的配置文件,让配置文件里面内容生效

b方式:传统 添加组件方式
先创建一个spring的配置文件xml:
在这里插入图片描述
编写xml:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
    <bean id="helloService" class="com.yhhy.springboot.service.HelloService"></bean>
</beans>

以上还是用以前的方式;
创建service:
在这里插入图片描述
编写测试类:

package com.yhhy.springboot;
import com.yhhy.springboot.bean.Person;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.context.ApplicationContext;
import org.springframework.test.context.junit4.SpringRunner;

@RunWith(SpringRunner.class)
@SpringBootTest
public class SpringBootDomeApplicationTests {
	@Autowired
	Person person;
	//将ioc容器让单元测试注入进来,这样就可以拿到了
	@Autowired
	ApplicationContext ioc;

	//测试容器中是否有helloService
	@Test
	public void testHelloService(){
		//是否包含,返回ture和false
		boolean b =  ioc.containsBean("helloService");
		System.out.println(b);
	}
//	@Test
//	public void contextLoads() {
//		System.out.println(person);
//	}

}

在这里插入图片描述
Spring Boot里面没有Spring的配置文件,我们自己编写的配置文件,也不能自动识别;
想让Spring的配置文件生效,加载进来;@ImportResource标注在一个配置类上
在这里插入图片描述
测试结果为true:
在这里插入图片描述

//导入Spring的配置文件让其生效
@ImportResource(locations = {"classpath:beans.xml"})
c、@Bean 注解:SpringBoot 添加组件配置类的方式

b方式在开发的时候太麻烦,需要写配置文件,还得在启动类上配置加载进去。
SpringBoot推荐给容器中添加组件的方式:

  • 不来编写配置文件了,推荐使用全注解的方式
  • 配置类==Spring配置文件
    1)、 创建配置类,配置要注入的类
package com.yhhy.springboot.springboot.config;
import com.yhhy.springboot.springboot.service.HelloService;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
/**
 * @Configuration: 指明当前类是一个配置类,就是来替代之前的Spring配置文件
 * spring配置时:<bean></bean>标签添加组件的
 */
@Configuration
public class MyAppConfig {
    /**
     * Bean注解:能标注在方法上,也能作为原注解;
     * 将方法的返回值添加到容器中,容器中这个组件默认的id就是方法名
     * @return
     */
    @Bean
    public HelloService helloService(){
        System.out.println("配置类@Bean给容器中添加组件了");
        return new HelloService();
    }
}

图例:在这里插入图片描述
HelloService图例:
在这里插入图片描述
运行测试结果:
在这里插入图片描述
可以看出来,此service是注入到容器中了,而且不用写第一种方法的xml和在主配置类上也不用加导入注解了

注意事项
SpringBoot 默认使用方法名作为类存放到容器的ID,这点要注意,例如我要改了方法名称后,获取需用最新的方法名;
在这里插入图片描述
在这里插入图片描述

properties配置文件 - 占位符

1、随机数
${random.value}、${random.int}、${random.long}
${random.int(10)}、${random.int[1024,65536]}

yml和properties都支持以下占位符的写法
在这里插入图片描述

  1. Person类
    在这里插入图片描述
  2. 配置类application.properties
    在这里插入图片描述
  3. 测试结果
    在这里插入图片描述
2、占位符获取之前配置的值,如果没有可以是用:指定默认值
person.last-name=张三${random.uuid}
person.age=${random.int}
person.birth=2017/12/15
person.boss=false
person.maps.k1=v1
person.maps.k2=14
person.lists=a,b,c
person.dog.name=${person.hello:hello}_dog
person.dog.age=15

p e r s o n . h e l l o : p e r s o n . h e l l o {person.hello} : 如果上面没有声名person.hello 就会输出 {person.hello}
${person.hello:hello} : 如果上面没有声名person.hello 就会输出默认值hello,也就是冒号后面的值

4、Profile 文件

Profile是spring对不同环境提供不同配置功能的支持(开发、测试、生产),可以通过激活指定参数等方式快读切换环境。

1、多Profile文件

我们在主配置文件编写的时候,文件名可以是application-{profile}.properties/yml,然后就可以动态来切换;
在这里插入图片描述

默认使用application.proerties ,可以激活使用相关环境配置文件

2、yml支持多文档方式

yml中的文档块方式:
在这里插入图片描述

3、激活指定profile

  1. 在配置文件properties中指定配置 spring.profiles.active=dev
  2. 命令行方式激活

4、激活方式详情

在这里插入图片描述

1)、命令行 方式1 激活配置

在这里插入图片描述
在这里插入图片描述
查看结果:
在这里插入图片描述

2)、命令行 方式2 激活配置
  1. 先打成jar包
    SpringBoot学习之路(二)- 输出HelloWorld及部署打包
  2. 使用命令启动
--spring.profiles.active=dev
  1. 测试结果
    在这里插入图片描述
    在这里插入图片描述
3)、配置文件 方式激活

在这里插入图片描述

4)、虚拟机参数 方式激活
-Dspring.profiles.active=dev

在这里插入图片描述

二、加载顺序

记住3点
SpringBoot也可以从以下位置加载配置;
优先级从高到低
高优先级的配置覆盖低优先级的配置,所有的配置会形成互补配置

1.命令行参数
所有的配置都可以在命令行上进行指定
java -jar spring-boot-02-config-02-0.0.1-SNAPSHOT.jar --server.port=8087 --server.context-path=/abc
多个配置用空格分开; --配置项=值
在这里插入图片描述
2.来自java:comp/env的JNDI属性
3.Java系统属性(System.getProperties())
4.操作系统环境变量
5.RandomValuePropertySource配置的random.*属性值

由jar包外向jar包内进行寻找;
优先加载带profile
6.jar包外部的application-{profile}.properties或application.yml(带spring.profile)配置文件
7.jar包内部的application-{profile}.properties或application.yml(带spring.profile)配置文件

再来加载不带profile
8.jar包外部的application.properties或application.yml(不带spring.profile)配置文件
9.jar包内部的application.properties或application.yml(不带spring.profile)配置文件
10.@Configuration注解类上的@PropertySource
11.通过SpringApplication.setDefaultProperties指定的默认属性
所有支持的配置加载来源:
配置文件能配置的属性参考官方文档
优先加载带profile 举例子:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

在这里插入图片描述
这是证明已经加载外部的配置了,但是切记如果代码中有指定profile,将不会加载外部资源,请记住一点:内部没有profile时才会加载外部资源

三、自动配置原理

自动配置原理

1)、SpringBoot启动的时候加载主配置类,开启了自动配置功能@EnableAutoConfiguration
2)

  • @EnableAutoConfiguration作用:
    点进去之后:利用@Import(EnaableAutoConfigurationImportSelector.class)给容器导入了一些;
  • EnaableAutoConfigurationImportSelector.class作用:
    点进入之后,查看此类的父类,父类规定了一个方法:可以查看selectImports方法中:
List<String> configurations = getCandidateConfigurations(annotationMetadata,      attributes)

获取候选的配置。
getCandidateConfigurations点进去之后:

SpringFactoriesLoader.loadFactoryNames()
扫描所有jar包类路径下  META-INF/spring.factories
把扫描到的这些文件的内容包装成properties对象
从properties中获取到EnableAutoConfiguration.class类(类名)对应的值,然后把他们添加在容器中

将 类路径下 META-INF/spring.factories 里面配置的所有EnableAutoConfiguration的值加入到了容器中;
有这些类添加到容器中:

# Auto Configure
org.springframework.boot.autoconfigure.EnableAutoConfiguration=\
org.springframework.boot.autoconfigure.admin.SpringApplicationAdminJmxAutoConfiguration,\
org.springframework.boot.autoconfigure.aop.AopAutoConfiguration,\
org.springframework.boot.autoconfigure.amqp.RabbitAutoConfiguration,\
org.springframework.boot.autoconfigure.batch.BatchAutoConfiguration,\
org.springframework.boot.autoconfigure.cache.CacheAutoConfiguration,\
org.springframework.boot.autoconfigure.cassandra.CassandraAutoConfiguration,\
org.springframework.boot.autoconfigure.cloud.CloudAutoConfiguration,\
org.springframework.boot.autoconfigure.context.ConfigurationPropertiesAutoConfiguration,\
org.springframework.boot.autoconfigure.context.MessageSourceAutoConfiguration,\
org.springframework.boot.autoconfigure.context.PropertyPlaceholderAutoConfiguration,\
org.springframework.boot.autoconfigure.couchbase.CouchbaseAutoConfiguration,\
org.springframework.boot.autoconfigure.dao.PersistenceExceptionTranslationAutoConfiguration,\
org.springframework.boot.autoconfigure.data.cassandra.CassandraDataAutoConfiguration,\
org.springframework.boot.autoconfigure.data.cassandra.CassandraRepositoriesAutoConfiguration,\
org.springframework.boot.autoconfigure.data.couchbase.CouchbaseDataAutoConfiguration,\
org.springframework.boot.autoconfigure.data.couchbase.CouchbaseRepositoriesAutoConfiguration,\
org.springframework.boot.autoconfigure.data.elasticsearch.ElasticsearchAutoConfiguration,\
org.springframework.boot.autoconfigure.data.elasticsearch.ElasticsearchDataAutoConfiguration,\
org.springframework.boot.autoconfigure.data.elasticsearch.ElasticsearchRepositoriesAutoConfiguration,\
org.springframework.boot.autoconfigure.data.jpa.JpaRepositoriesAutoConfiguration,\
org.springframework.boot.autoconfigure.data.ldap.LdapDataAutoConfiguration,\
org.springframework.boot.autoconfigure.data.ldap.LdapRepositoriesAutoConfiguration,\
org.springframework.boot.autoconfigure.data.mongo.MongoDataAutoConfiguration,\
org.springframework.boot.autoconfigure.data.mongo.MongoRepositoriesAutoConfiguration,\
org.springframework.boot.autoconfigure.data.neo4j.Neo4jDataAutoConfiguration,\
org.springframework.boot.autoconfigure.data.neo4j.Neo4jRepositoriesAutoConfiguration,\
org.springframework.boot.autoconfigure.data.solr.SolrRepositoriesAutoConfiguration,\
org.springframework.boot.autoconfigure.data.redis.RedisAutoConfiguration,\
org.springframework.boot.autoconfigure.data.redis.RedisRepositoriesAutoConfiguration,\
org.springframework.boot.autoconfigure.data.rest.RepositoryRestMvcAutoConfiguration,\
org.springframework.boot.autoconfigure.data.web.SpringDataWebAutoConfiguration,\
org.springframework.boot.autoconfigure.elasticsearch.jest.JestAutoConfiguration,\
org.springframework.boot.autoconfigure.freemarker.FreeMarkerAutoConfiguration,\
org.springframework.boot.autoconfigure.gson.GsonAutoConfiguration,\
org.springframework.boot.autoconfigure.h2.H2ConsoleAutoConfiguration,\
org.springframework.boot.autoconfigure.hateoas.HypermediaAutoConfiguration,\
org.springframework.boot.autoconfigure.hazelcast.HazelcastAutoConfiguration,\
org.springframework.boot.autoconfigure.hazelcast.HazelcastJpaDependencyAutoConfiguration,\
org.springframework.boot.autoconfigure.info.ProjectInfoAutoConfiguration,\
org.springframework.boot.autoconfigure.integration.IntegrationAutoConfiguration,\
org.springframework.boot.autoconfigure.jackson.JacksonAutoConfiguration,\
org.springframework.boot.autoconfigure.jdbc.DataSourceAutoConfiguration,\
org.springframework.boot.autoconfigure.jdbc.JdbcTemplateAutoConfiguration,\
org.springframework.boot.autoconfigure.jdbc.JndiDataSourceAutoConfiguration,\
org.springframework.boot.autoconfigure.jdbc.XADataSourceAutoConfiguration,\
org.springframework.boot.autoconfigure.jdbc.DataSourceTransactionManagerAutoConfiguration,\
org.springframework.boot.autoconfigure.jms.JmsAutoConfiguration,\
org.springframework.boot.autoconfigure.jmx.JmxAutoConfiguration,\
org.springframework.boot.autoconfigure.jms.JndiConnectionFactoryAutoConfiguration,\
org.springframework.boot.autoconfigure.jms.activemq.ActiveMQAutoConfiguration,\
org.springframework.boot.autoconfigure.jms.artemis.ArtemisAutoConfiguration,\
org.springframework.boot.autoconfigure.flyway.FlywayAutoConfiguration,\
org.springframework.boot.autoconfigure.groovy.template.GroovyTemplateAutoConfiguration,\
org.springframework.boot.autoconfigure.jersey.JerseyAutoConfiguration,\
org.springframework.boot.autoconfigure.jooq.JooqAutoConfiguration,\
org.springframework.boot.autoconfigure.kafka.KafkaAutoConfiguration,\
org.springframework.boot.autoconfigure.ldap.embedded.EmbeddedLdapAutoConfiguration,\
org.springframework.boot.autoconfigure.ldap.LdapAutoConfiguration,\
org.springframework.boot.autoconfigure.liquibase.LiquibaseAutoConfiguration,\
org.springframework.boot.autoconfigure.mail.MailSenderAutoConfiguration,\
org.springframework.boot.autoconfigure.mail.MailSenderValidatorAutoConfiguration,\
org.springframework.boot.autoconfigure.mobile.DeviceResolverAutoConfiguration,\
org.springframework.boot.autoconfigure.mobile.DeviceDelegatingViewResolverAutoConfiguration,\
org.springframework.boot.autoconfigure.mobile.SitePreferenceAutoConfiguration,\
org.springframework.boot.autoconfigure.mongo.embedded.EmbeddedMongoAutoConfiguration,\
org.springframework.boot.autoconfigure.mongo.MongoAutoConfiguration,\
org.springframework.boot.autoconfigure.mustache.MustacheAutoConfiguration,\
org.springframework.boot.autoconfigure.orm.jpa.HibernateJpaAutoConfiguration,\
org.springframework.boot.autoconfigure.reactor.ReactorAutoConfiguration,\
org.springframework.boot.autoconfigure.security.SecurityAutoConfiguration,\
org.springframework.boot.autoconfigure.security.SecurityFilterAutoConfiguration,\
org.springframework.boot.autoconfigure.security.FallbackWebSecurityAutoConfiguration,\
org.springframework.boot.autoconfigure.security.oauth2.OAuth2AutoConfiguration,\
org.springframework.boot.autoconfigure.sendgrid.SendGridAutoConfiguration,\
org.springframework.boot.autoconfigure.session.SessionAutoConfiguration,\
org.springframework.boot.autoconfigure.social.SocialWebAutoConfiguration,\
org.springframework.boot.autoconfigure.social.FacebookAutoConfiguration,\
org.springframework.boot.autoconfigure.social.LinkedInAutoConfiguration,\
org.springframework.boot.autoconfigure.social.TwitterAutoConfiguration,\
org.springframework.boot.autoconfigure.solr.SolrAutoConfiguration,\
org.springframework.boot.autoconfigure.thymeleaf.ThymeleafAutoConfiguration,\
org.springframework.boot.autoconfigure.transaction.TransactionAutoConfiguration,\
org.springframework.boot.autoconfigure.transaction.jta.JtaAutoConfiguration,\
org.springframework.boot.autoconfigure.validation.ValidationAutoConfiguration,\
org.springframework.boot.autoconfigure.web.DispatcherServletAutoConfiguration,\
org.springframework.boot.autoconfigure.web.EmbeddedServletContainerAutoConfiguration,\
org.springframework.boot.autoconfigure.web.ErrorMvcAutoConfiguration,\
org.springframework.boot.autoconfigure.web.HttpEncodingAutoConfiguration,\
org.springframework.boot.autoconfigure.web.HttpMessageConvertersAutoConfiguration,\
org.springframework.boot.autoconfigure.web.MultipartAutoConfiguration,\
org.springframework.boot.autoconfigure.web.ServerPropertiesAutoConfiguration,\
org.springframework.boot.autoconfigure.web.WebClientAutoConfiguration,\
org.springframework.boot.autoconfigure.web.WebMvcAutoConfiguration,\
org.springframework.boot.autoconfigure.websocket.WebSocketAutoConfiguration,\
org.springframework.boot.autoconfigure.websocket.WebSocketMessagingAutoConfiguration,\
org.springframework.boot.autoconfigure.webservices.WebServicesAutoConfiguration

每一个这样的 xxxAutoConfiguration类都是容器中的一个组件,都加入到容器中,用他们来做自动配置;
3)、每一个自动配置类进行自动配置功能;
4)、以上面列表中的:HttpEncodingAutoConfiguration为例解释自动配置原理;
点入此类中 - HttpEncodingAutoConfiguration :

@Configuration   //表示这是一个配置类,以前编写的配置文件一样,也可以给容器中添加组件
@EnableConfigurationProperties(HttpEncodingProperties.class)  //启动指定类的ConfigurationProperties功能;将配置文件中对应的值和HttpEncodingProperties绑定起来;并把HttpEncodingProperties加入到ioc容器中

@ConditionalOnWebApplication //Spring底层@Conditional注解(Spring注解版),根据不同的条件,如果满足指定的条件,整个配置类里面的配置就会生效;    判断当前应用是否是web应用,如果是,当前配置类生效

@ConditionalOnClass(CharacterEncodingFilter.class)  //判断当前项目有没有这个类CharacterEncodingFilter;SpringMVC中进行乱码解决的过滤器;

@ConditionalOnProperty(prefix = "spring.http.encoding", value = "enabled", matchIfMissing = true)  //判断配置文件中是否存在某个配置  spring.http.encoding.enabled;如果不存在,判断也是成立的
//即使我们配置文件中不配置pring.http.encoding.enabled=true,也是默认生效的;
public class HttpEncodingAutoConfiguration {
  
  	//他已经和SpringBoot的配置文件映射了
  	private final HttpEncodingProperties properties;
  
   //只有一个有参构造器的情况下,参数的值就会从容器中拿
  	public HttpEncodingAutoConfiguration(HttpEncodingProperties properties) {
		this.properties = properties;
	}
  
    @Bean   //给容器中添加一个组件,这个组件的某些值需要从properties中获取
	@ConditionalOnMissingBean(CharacterEncodingFilter.class) //判断容器没有这个组件?
	public CharacterEncodingFilter characterEncodingFilter() {
		CharacterEncodingFilter filter = new OrderedCharacterEncodingFilter();
		filter.setEncoding(this.properties.getCharset().name());
		filter.setForceRequestEncoding(this.properties.shouldForce(Type.REQUEST));
		filter.setForceResponseEncoding(this.properties.shouldForce(Type.RESPONSE));
		return filter;
	}

根据当前不同的条件判断,决定这个配置类是否生效?

一但这个配置类生效;这个配置类就会给容器中添加各种组件;这些组件的属性是从对应的properties类中获取的,这些类里面的每一个属性又是和配置文件绑定的,所以说:我们在配置文件中能配那些属性,主要看这个配置类;

SpringBoot精髓:
1)、SpringBoot启动会加载大量的自动配置类

2)、我们看我们需要的功能有没有SpringBoot默认写好的自动配置类;

3)、我们再来看这个自动配置类中到底配置了哪些组件;(只要我们要用的组件有,我们就不需要再来配置了)

4)、给容器中自动配置类添加组件的时候,会从properties类中获取某些属性。我们就可以在配置文件中指定这些属性的值;

猜你喜欢

转载自blog.csdn.net/qq_36698956/article/details/85994093