核心功能-原理解析
核心功能-原理解析
John Doe原理解析
1. Profile功能
为了方便多环境适配,springboot简化了profile功能。
1. application-profile功能
默认配置文件 application.yaml;任何时候都会加载
指定环境配置文件
application-{env}.yaml激活指定环境
- 配置文件激活
- 命令行激活:
java -jar xxx.jar --spring.profiles.active=prod --person.name=haha
- 修改配置文件的任意值,命令行优先
默认配置与环境配置同时生效
同名配置项,profile配置优先
2. @Profile条件装配功能
1 |
|
3. profile分组
1 | spring.profiles.group.production[0]=proddb |
2. 外部化配置
- Default properties (specified by setting
SpringApplication.setDefaultProperties). @PropertySourceannotations on your@Configurationclasses. Please note that such property sources are not added to theEnvironmentuntil the application context is being refreshed. This is too late to configure certain properties such aslogging.*andspring.main.*which are read before refresh begins.- Config data (such as
application.propertiesfiles) - A
RandomValuePropertySourcethat has properties only inrandom.*. - OS environment variables.
- Java System properties (
System.getProperties()). - JNDI attributes from
java:comp/env. ServletContextinit parameters.ServletConfiginit parameters.- Properties from
SPRING_APPLICATION_JSON(inline JSON embedded in an environment variable or system property). - Command line arguments.
propertiesattribute on your tests. Available on@SpringBootTestand the test annotations for testing a particular slice of your application.@TestPropertySourceannotations on your tests.- Devtools global settings properties in the
$HOME/.config/spring-bootdirectory when devtools is active.
1. 外部配置源
常用:Java属性文件、YAML文件、环境变量、命令行参数;
2. 配置文件查找位置
(1) classpath 根路径
(2) classpath 根路径下config目录
(3) jar包当前目录
(4) jar包当前目录的config目录
(5) /config子目录的直接子目录
3. 配置文件加载顺序
- 当前jar包内部的
application.properties和application.yml - 当前jar包内部的
application-{profile}.properties和application-{profile}.yml - 引用的外部jar包的
application.properties和application.yml - 引用的外部jar包的
application-{profile}.properties和application-{profile}.yml
3. 自定义starter
1. starter启动原理
- starter-pom引入 autoconfigurer 包
autoconfigure包中配置使用 META-INF/spring.factories 中 EnableAutoConfiguration 的值,使得项目启动加载指定的自动配置类
编写自动配置类 xxxAutoConfiguration -> xxxxProperties
@Configuration
@Conditional
@EnableConfigurationProperties
@Bean
引入starter — xxxAutoConfiguration — 容器中放入组件 —- 绑定xxxProperties —- 配置项
2. 自定义starter
atguigu-hello-spring-boot-starter(启动器)
atguigu-hello-spring-boot-starter-autoconfigure(自动配置包)
4. SpringBoot原理
Spring原理【Spring注解】、SpringMVC原理、自动配置原理、SpringBoot原理
1. SpringBoot启动过程
创建
SpringApplication保存一些信息。
判定当前应用的类型。ClassUtils –> Servlet
bootstrappers:初始启动引导器(List<Bootstrapper>):去spring.factories文件中找org.springframework.boot.Bootstrapper1
2
3
4
5
6
7
8
9public interface Bootstrapper {
/**
* Initialize the given {@link BootstrapRegistry} with any required registrations.
* @param registry the registry to initialize
*/
void intitialize(BootstrapRegistry registry);
}ApplicationContextInitializer:去spring.factories找 ApplicationContextInitializerList<ApplicationContextInitializer<?>> initializers
ApplicationListener:应用监听器。去spring.factories找 ApplicationListenerList<ApplicationListener<?>> listeners
运行
SpringApplication记录应用的启动时间
创建引导上下文(Context环境)
createBootstrapContext()- 获取到所有之前的
bootstrappers挨个执行intitialize()来完成对引导启动器上下文环境设置
- 获取到所有之前的
让当前应用进入
headless模式。java.awt.headless获取所有
RunListener(运行监听器)【为了方便所有Listener进行事件感知】getSpringFactoriesInstances去spring.factories找SpringApplicationRunListener
遍历
SpringApplicationRunListener调用 starting 方法;- 相当于通知所有感兴趣系统正在启动过程的人,项目正在
starting。
- 相当于通知所有感兴趣系统正在启动过程的人,项目正在
保存命令行参数;
ApplicationArguments准备环境
prepareEnvironment();- 返回或者创建基础环境信息对象。StandardServletEnvironment
- 配置环境信息对象。
- 读取所有的配置源的配置属性值。
- 绑定环境信息
- 监听器调用
listener.environmentPrepared();通知所有的监听器当前环境准备完成
创建IOC容器(
createApplicationContext())- 根据项目类型(Servlet)创建容器,
- 当前会创建
AnnotationConfigServletWebServerApplicationContext
准备ApplicationContext IOC容器的基本信息
prepareContext()- 保存环境信息
- IOC容器的后置处理流程。
- 应用初始化器;
applyInitializers;- 遍历所有的
ApplicationContextInitializer。调用 initialize。来对ioc容器进行初始化扩展功能 - 遍历所有的 listener 调用
contextPrepared。EventPublishRunListenr;通知所有的监听器contextPrepared
- 遍历所有的
- 所有的监听器调用 contextLoaded。通知所有的监听器 contextLoaded;
刷新IOC容器。refreshContext
- 创建容器中的所有组件(Spring注解)
容器刷新完成后工作?afterRefresh
所有监听器调用
listeners.started(context); 通知所有的监听器 started调用所有runners;
callRunners()获取容器中的
ApplicationRunner1
2
3
4
5
6
7
8
9
10
11
public interface ApplicationRunner {
/**
* Callback used to run the bean.
* @param args incoming application arguments
* @throws Exception on error
*/
void run(ApplicationArguments args) throws Exception;
}获取容器中的
CommandLineRunner1
2
3
4
5
6
7
8
9
10
11
public interface CommandLineRunner {
/**
* Callback used to run the bean.
* @param args incoming main method arguments
* @throws Exception on error
*/
void run(String... args) throws Exception;
}合并所有runner并且按照@Order进行排序
遍历所有的runner。调用 run 方法
如果以上有异常
- 调用Listener 的
failed
- 调用Listener 的
调用所有监听器的 running 方法
listeners.running(context); 通知所有的监听器 runningrunning如果有问题。继续通知
failed。调用所有 Listener 的 failed;通知所有的监听器 failed
2. ApplicationContextInitializer
1 | # spring.factories中配置 |
1 | public class MyApplicationContextInitializer implements ApplicationContextInitializer { |
3. ApplicationListener
1 | # spring.factories中配置 |
1 | public class MyApplicationListener implements ApplicationListener { |
4. SpringApplicationRunListener
1 | # spring.factories中配置 |
1 | public class MySpringApplicationRunListener implements SpringApplicationRunListener { |
5. ApplicationRunner
1 |
|
6. CommandLineRunner
1 | /** |


