【Spring Boot 四】启动之准备系统环境environmentPrepared

简介: 【Spring Boot 四】启动之准备系统环境environmentPrepared

SpringBoot启动的时候 listeners.starting() ;接下来就是准备环境的过程

environmentPrepared

系统环境已经准备就绪

  private ConfigurableEnvironment prepareEnvironment(
      SpringApplicationRunListeners listeners,
      ApplicationArguments applicationArguments) {
    // Create and configure the environment
    ConfigurableEnvironment environment = getOrCreateEnvironment();
    configureEnvironment(environment, applicationArguments.getSourceArgs());
    listeners.environmentPrepared(environment);
    bindToSpringApplication(environment);
    if (!this.isCustomEnvironment) {
      environment = new EnvironmentConverter(getClassLoader())
          .convertEnvironmentIfNecessary(environment, deduceEnvironmentClass());
    }
    ConfigurationPropertySources.attach(environment);
    return environment;
  }

上面代码总结来说分为下面几步

  • 创建 ConfigurableEnvironment ; 增加至少(根据启动类型不同,可能还会增加其他的属性源)两个属性源 一个Jvm属性源;一个环境变量属性源
propertySources.addLast(new MapPropertySource(SYSTEM_PROPERTIES_PROPERTY_SOURCE_NAME, getSystemProperties()));
propertySources.addLast(new SystemEnvironmentPropertySource(SYSTEM_ENVIRONMENT_PROPERTY_SOURCE_NAME, getSystemEnvironment()))

增加命令行属性源 (添加到最前面,优先级最高); 就是启动时候的那个入参Args;详情看SpringBoot 一 SpringApplication启动类的Args详解

sources.addFirst(new SimpleCommandLinePropertySource(args));

触发listeners.environmentPrepared(environment)事件

bindToSpringApplication

上面几个步骤我们主要分析一下


环境准备就绪事件通知listeners.environmentPrepared(environment)

通过断点看到有下面几个监听者


image.png

image.png

选两个监听者分析

  • ConfigFileApplicationListener
  • DelegatingApplicationListener

ConfigFileApplicationListener

事件通知到之后 是执行了下面的方法

  private void onApplicationEnvironmentPreparedEvent(
      ApplicationEnvironmentPreparedEvent event) {
    List<EnvironmentPostProcessor> postProcessors = loadPostProcessors();
    postProcessors.add(this);
    AnnotationAwareOrderComparator.sort(postProcessors);
    for (EnvironmentPostProcessor postProcessor : postProcessors) {
      postProcessor.postProcessEnvironment(event.getEnvironment(),
          event.getSpringApplication());
    }
  }
  • 通过spring.factories方式加载EnvironmentPostProcessor实现类;
  • 自身也是一个EnvironmentPostProcessor
  • 将上述EnvironmentPostProcessor排序之后执行postProcessEnvironment方法

image.png

SystemEnvironmentPropertySourceEnvironmentPostProcessor

将之前加载的系统属性对象 替换陈给一个新的对象;但是属性; 这样做的原因TODO

image.png

image.png

ConfigFileApplicationListener

作为一个EnvironmentPostProcessor的时候,他的作用是想environment中添加了一个RandomValuePropertySource属性源; 可以通过environment.getProperty("random.*")返回各种随机值


RandomValuePropertySource用法

environment.getProperty("random.int")
environment.getProperty("random.long")
environment.getProperty("random.int.5,100;") 5~100中随机(后面的;要接上,因为它会截掉最后一个字符;)
environment.getProperty("random.long.5,10000;") 5~10000中随机(后面的;要接上,因为它会截掉最后一个字符;)
environment.getProperty("random.uuid")

new Loader(environment, resourceLoader).load(); 加载 配置文件中的属性

  Loader(ConfigurableEnvironment environment, ResourceLoader resourceLoader) {
      this.environment = environment;
      this.placeholdersResolver = new PropertySourcesPlaceholdersResolver(
          this.environment);
      this.resourceLoader = (resourceLoader != null) ? resourceLoader
          : new DefaultResourceLoader();
      this.propertySourceLoaders = SpringFactoriesLoader.loadFactories(
          PropertySourceLoader.class, getClass().getClassLoader());
    }

image.png

配置文件加载器有两个 他们都 实现了 接口 PropertySourceLoader


PropertiesPropertySourceLoader 解析 . properties 和.xml文件

YamlPropertySourceLoader 解析 .yml 和 .yaml文件

加载load()

    public void load() {
      this.profiles = new LinkedList<>();
      this.processedProfiles = new LinkedList<>();
      this.activatedProfiles = false;
      this.loaded = new LinkedHashMap<>();
      initializeProfiles();
      while (!this.profiles.isEmpty()) {
        Profile profile = this.profiles.poll();
        if (profile != null && !profile.isDefaultProfile()) {
          addProfileToEnvironment(profile.getName());
        }
        load(profile, this::getPositiveProfileFilter,
            addToLoaded(MutablePropertySources::addLast, false));
        this.processedProfiles.add(profile);
      }
      resetEnvironmentProfiles(this.processedProfiles);
      load(null, this::getNegativeProfileFilter,
          addToLoaded(MutablePropertySources::addFirst, true));
      addLoadedPropertySources();
    }

简单概述一下这个Loader.load()方法做了什么

1.初始化配置文件 initializeProfiles; 读取属性spring.profiles.include; spring.profiles.active的值添加到使用的配置文件属性

2. 将1读取到的配置文件解析加载属性到Environment中


image.png

image.png

看图中最后两个就是属性源就是加载完配置文件后添加的


Binder


简单来说,就是将Environment中的属性值,绑定到某个对象中去的;比如SpringApplication中的属性bannerMode 默认是 Banner.Mode.CONSOLE 但是我在配置文件中spring.main.banner-mode=log 将它改成log形式,为啥修改可以成功呢?是因为在启动过程中执行了下面的代码

image.png

他会将spring.main开头的配置都会绑定到 Bindable.ofInstance(this)中 这个this就是SpringApplication


Springboot 2.x新引入的类,负责处理对象与多个ConfigurationPropertySource(属性)之间的绑定。

比Environment类好用很多,可以非常方便地进行类型转换,以及提供回调方法介入绑定的各个阶段进行深度定制。

以前获取属性值是 用Environment中的方法,现

  # 判断是否包含键值
  boolean containsProperty(String key);
  # 获取属性值,如果获取不到返回null
  String getProperty(String key);
  # 获取属性值,如果获取不到返回缺省值
  String getProperty(String key, String defaultValue);
  # 获取属性对象;其转换和Converter有关,会根据sourceType和targetType查找转换器
  <T> T getProperty(String key, Class<T> targetType)

现在用

SpringBoot属性绑定Environment和Binder

image.png

DelegatingApplicationListener

从环境中读取属性 context.listener.classes;的值 并且将它们都实例化; 这些calss必须是ApplicationListener的实现类; 实例化好了之后加入到监听器列表;这是另一种实现 自定义监听与通知的 方式; SpringBoot 自定义监听与通知


bindToSpringApplication

将 environment中的spring.main开头的属性 绑定到SpringApplication 中的属性值上

protected void bindToSpringApplication(ConfigurableEnvironment environment) {
    try {
      Binder.get(environment).bind("spring.main", Bindable.ofInstance(this));
    }
    catch (Exception ex) {
      throw new IllegalStateException("Cannot bind to SpringApplication", ex);
    }
  }

image.png


相关文章
|
4月前
|
Cloud Native Java 对象存储
面向未来的架构设计:Spring Cloud和Netflix OSS在云原生环境下的发展趋势
展望未来,随着5G、边缘计算等新技术的兴起,微服务架构的设计理念将会更加深入人心,Spring Cloud和Netflix OSS也将继续引领技术潮流,为企业带来更为高效、灵活且强大的解决方案。无论是对于初创公司还是大型企业而言,掌握这些前沿技术都将是在激烈市场竞争中脱颖而出的关键所在。
75 0
|
28天前
|
Java 数据库 数据安全/隐私保护
轻松掌握Spring依赖注入:打造你的登录验证系统
本文以轻松活泼的风格,带领读者走进Spring框架中的依赖注入和登录验证的世界。通过详细的步骤和代码示例,我们从DAO层的创建到Service层的实现,再到Spring配置文件的编写,最后通过测试类验证功能,一步步构建了一个简单的登录验证系统。文章不仅提供了实用的技术指导,还以口语化和生动的语言,让学习变得不再枯燥。
40 2
|
2月前
|
存储 运维 安全
Spring运维之boot项目多环境(yaml 多文件 proerties)及分组管理与开发控制
通过以上措施,可以保证Spring Boot项目的配置管理在专业水准上,并且易于维护和管理,符合搜索引擎收录标准。
52 2
|
2月前
|
XML Java 数据库连接
SpringBoot集成Flowable:打造强大的工作流管理系统
在企业级应用开发中,工作流管理是一个核心组件,它能够帮助我们定义、执行和管理业务流程。Flowable是一个开源的工作流和业务流程管理(BPM)平台,它提供了强大的工作流引擎和建模工具。结合SpringBoot,我们可以快速构建一个高效、灵活的工作流管理系统。本文将探讨如何将Flowable集成到SpringBoot应用中,并展示其强大的功能。
346 1
|
3月前
|
SQL JSON Java
mybatis使用三:springboot整合mybatis,使用PageHelper 进行分页操作,并整合swagger2。使用正规的开发模式:定义统一的数据返回格式和请求模块
这篇文章介绍了如何在Spring Boot项目中整合MyBatis和PageHelper进行分页操作,并且集成Swagger2来生成API文档,同时定义了统一的数据返回格式和请求模块。
92 1
mybatis使用三:springboot整合mybatis,使用PageHelper 进行分页操作,并整合swagger2。使用正规的开发模式:定义统一的数据返回格式和请求模块
|
2月前
|
JavaScript Java 项目管理
Java毕设学习 基于SpringBoot + Vue 的医院管理系统 持续给大家寻找Java毕设学习项目(附源码)
基于SpringBoot + Vue的医院管理系统,涵盖医院、患者、挂号、药物、检查、病床、排班管理和数据分析等功能。开发工具为IDEA和HBuilder X,环境需配置jdk8、Node.js14、MySQL8。文末提供源码下载链接。
|
3月前
|
存储 安全 Java
打造智能合同管理系统:SpringBoot与电子签章的完美融合
【10月更文挑战第7天】 在数字化转型的浪潮中,电子合同管理系统因其高效、环保和安全的特点,正逐渐成为企业合同管理的新宠。本文将分享如何利用SpringBoot框架实现一个集电子文件签字与合同管理于一体的智能系统,探索技术如何助力合同管理的现代化。
137 4
|
3月前
|
前端开发 Java Apache
SpringBoot实现电子文件签字+合同系统!
【10月更文挑战第15天】 在现代企业运营中,合同管理和电子文件签字成为了日常活动中不可或缺的一部分。随着技术的发展,电子合同系统因其高效性、安全性和环保性,逐渐取代了传统的纸质合同。本文将详细介绍如何使用SpringBoot框架实现一个电子文件签字和合同管理系统。
137 1
|
4月前
|
前端开发 JavaScript Java
基于Java+Springboot+Vue开发的大学竞赛报名管理系统
基于Java+Springboot+Vue开发的大学竞赛报名管理系统(前后端分离),这是一项为大学生课程设计作业而开发的项目。该系统旨在帮助大学生学习并掌握Java编程技能,同时锻炼他们的项目设计与开发能力。通过学习基于Java的大学竞赛报名管理系统项目,大学生可以在实践中学习和提升自己的能力,为以后的职业发展打下坚实基础。
242 3
基于Java+Springboot+Vue开发的大学竞赛报名管理系统
|
3月前
|
文字识别 安全 Java
SpringBoot3.x和OCR构建车牌识别系统
本文介绍了一个基于Java SpringBoot3.x框架的车牌识别系统,详细阐述了系统的设计目标、需求分析及其实现过程。利用Tesseract OCR库和OpenCV库,实现了车牌图片的识别与处理,确保系统的高准确性和稳定性。文中还提供了具体的代码示例,展示了如何构建和优化车牌识别服务,以及如何处理特殊和异常车牌。通过实际应用案例,帮助读者理解和应用这一解决方案。