Spring Cloud项目Application Run两次秘密解析
2021-03-03 21:56
281 查看
ApplicationContextInitializer介绍
作用是在
ConfigurableApplicationContext类型的
ApplicationContext.
refresh操作 之前,允许我们对
ConfiurableApplicationContext增强处理的扩展。
业务场景
在实际开发过程中,web应用中需要编程方式对应用上下文做初始化。比如,注册属性源
(bootstrap/application properties sources);编码动态激活不同
profile对应
environment
最近项目,要根据不同的环境
Linux/Windows来加载不同的SDK 参数.
自定义Condition
实现
由于配置项极其的多,通过修改
ConfigurationPropertiesBean上自定义
@Conditional
public class LinuxCondition implements Condition{ @Override public boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata) { String property = context.getEnvironment().getProperty("os.name"); ... return property.contains("linux"); }}
@Target({ElementType.TYPE, ElementType.METHOD})@Retention(RetentionPolicy.RUNTIME)@Documented@Conditional({LinuxCondition.class})public @interface ConditionOnLinux {}
由于SDK 设计的配置类抽取的太多,拆分的不太合理,在不破坏文物的情况下所以放弃了这种方式
自定义 ApplicationContextInitializer 实现
根据上下文环境,加载
resource目录,不同环境的配置文件
public class SelectApplicationContextInitializer implements ApplicationContextInitializer { @Override public void initialize(ConfigurableApplicationContext context) { ConfigurableEnvironment env = context.getEnvironment(); MutablePropertySources mps = env.getPropertySources(); String property = env.getProperty("os.name"); if (property.contains("Mac OS X")) { mps.addLast(new ResourcePropertySource(new ClassPathResource("linux.properties"))); } else { mps.addLast(new ResourcePropertySource(new ClassPathResource("window.properties"))); } }}
问题: initialize 执行两次
相关功能抽取成
starter,运行在单体的
Spring Boot项目,若加入
Spring Cloud Context则会执行两次 上文代码
SpringApplication.run
- 为了找出问题真凶,在 SpringApplication run 方法下打上了断点。
BootstrapApplicationListener
一路跟到了
BootstrapApplicationListener.bootstrapServiceContext方法。
- 我们看看
bootstrapServiceContext
方法,
SpringApplicationBuilder builder = (new SpringApplicationBuilder(new Class[0])).profiles(environment.getActiveProfiles()).bannerMode(Mode.OFF).environment(bootstrapEnvironment).registerShutdownHook(false).logStartupInfo(false).web(WebApplicationType.NONE);SpringApplication builderApplication = builder.application();if (builderApplication.getMainApplicationClass() == null) { builder.main(application.getMainApplicationClass());}if (environment.getPropertySources().contains("refreshArgs")) { builderApplication.setListeners(this.filterListeners(builderApplication.getListeners()));}builder.sources(new Class[]{BootstrapImportSelectorConfiguration.class});ConfigurableApplicationContext context = builder.run(new String[0]);context.setId("bootstrap");
真相预警
BootstrapApplicationListener
里,利用 SpringApplicationBuilder
进行了一次重启, 虽然是Run
两次但是第一次 并未到 启动容器等,所以出现Bean
加载两次,或者 运行容器 Tomcat
等端口冲突。
相关文章推荐
- 解析$.grep()源码及透过$.grep()看(两次取反)!!的作用
- 深度解析 机器人自主移动的秘密
- 走近华佗,解析自动化故障处理系统背后的秘密
- jquery json xml关于解析两次json数据动态取值的处理方式
- 基础向:详细解析「财务对账」的秘密
- 9000字全面解析「分销平台」背后的秘密
- Chrome高性能的秘密:DNS预解析
- 你不知道的秘密——深入解析前缀,中缀,后缀表达式
- 编译器内部的秘密--微软的编译器是如何解析Try/Catch/Throw的
- 走近华佗,解析自动化故障处理系统背后的秘密
- 翻译:编译器内部的秘密--微软的编译器是如何解析Try/Catch/Throw的
- 【C解析之十二】C语言 printf 的秘密
- 【MYSQL】innodb两次写(double write)实现解析
- 解析软件系统稳定性的三大秘密
- 探索C++的秘密之详解extern "C",这就是为什么很多.lib被我们正确调用确总是无法解析的。
- 编译器内部的秘密--微软的编译器是如何解析
- 知乎 Matisse 源码解析,探究高效图片选择库的秘密
- (转)探索C++的秘密之详解extern "C",这就是为什么很多.lib被我们正确调用确总是无法解析的。
- 走近华佗,解析自动化故障处理系统背后的秘密
- 解析一段QQ对话背后的秘密