首页 > 其他分享 >Spring Environment 容器环境变量注入Spring属性不一致却能生效

Spring Environment 容器环境变量注入Spring属性不一致却能生效

时间:2024-06-20 10:32:45浏览次数:24  
标签:读取 return String Spring Environment 环境变量 name

https://blog.csdn.net/fenglllle/article/details/126942480

 

前言
最近使用容器部署应用,Spring应用,会注入一些环境变量,然而这些环境变量的大小写和真实的取值差异很大,而且也不是xxx.xxx,而是xxx_xxx,非常奇怪,代码里也没发现原因。通过分析Spring源码发现,原理就是Spring的特殊处理,以及Spring的设计。

1.注入系统环境变量
随意写一个demo

import org.junit.jupiter.api.Test;
import org.springframework.core.env.SystemEnvironmentPropertySource;

import java.util.HashMap;
import java.util.Map;

public class EnvPropertiesTest {

@Test
public void getProperties(){
Map<String, Object> map = new HashMap<>();
map.put("SPRING_DEF_DEMO", "hi, I`m a demo");
map.put("SPRING-DEF-DEMO", "hi, I`m a demo2222");
map.put("spring.def.demo", "hi, I`m a demo2222333");
SystemEnvironmentPropertySource propertySource = new SystemEnvironmentPropertySource("sysEnv", map);
System.out.println(propertySource.getProperty("spring.def.demo"));
}
}

运行后可以得出

 

跟我们想的一致,那么现在实现特殊操作,把

map.put("spring.def.demo", "hi, I`m a demo2222333");
注释掉

 

懵了,为啥获取spring.def.demo,却拿到了SPRING_DEF_DEMO

 

另外通过SPRING-DEF-DEMO也可以读取到 SPRING_DEF_DEMO的值。这就是容器注入大写下划线,而通过常规方式读取到的现象,实际上大写加下划线可以认为常量定义。

2.源码分析原理
原理实际上可以看到是Spring自己处理了。那么Spring是怎么处理的呢,Spring读取环境变量实际上和系统变量是有区别的。

org.springframework.core.env.StandardEnvironment

定义在Spring-core包中

protected void customizePropertySources(MutablePropertySources propertySources) {
propertySources.addLast(
new PropertiesPropertySource(SYSTEM_PROPERTIES_PROPERTY_SOURCE_NAME, getSystemProperties()));
propertySources.addLast(
new SystemEnvironmentPropertySource(SYSTEM_ENVIRONMENT_PROPERTY_SOURCE_NAME, getSystemEnvironment()));
}
上面的是系统变量,下面是环境变量

System.getenv()
System.getProperties()
上面的demo,之所以使用

SystemEnvironmentPropertySource
就是这里Spring定义决定的,关键在于getProperty方法

public Object getProperty(String name) {
String actualName = resolvePropertyName(name);
if (logger.isDebugEnabled() && !name.equals(actualName)) {
logger.debug("PropertySource '" + getName() + "' does not contain property '" + name +
"', but found equivalent '" + actualName + "'");
}
return super.getProperty(actualName);
}
核心在于

String actualName = resolvePropertyName(name);
真实名称,这就是Spring的环境变量的设计

protected final String resolvePropertyName(String name) {
Assert.notNull(name, "Property name must not be null");
// 拿到替换了. -为_的name读取,前提是name不能直接取到值数据
String resolvedName = checkPropertyName(name);
if (resolvedName != null) {
return resolvedName;
}
//大写处理,注入大写之所以可以读取,根源就是这里
String uppercasedName = name.toUpperCase();
if (!name.equals(uppercasedName)) { //严谨处理,毕竟eq就没必要进一步处理了
resolvedName = checkPropertyName(uppercasedName); //大写进一步看是否有大写属性值
if (resolvedName != null) {
return resolvedName;
}
}
return name;
}

继续

private String checkPropertyName(String name) {
// Check name as-is 如果能直接取到属性,就不转义
if (containsKey(name)) {
return name;
}
// Check name with just dots replaced
// 替换.为_
String noDotName = name.replace('.', '_');
if (!name.equals(noDotName) && containsKey(noDotName)) {
return noDotName;
}
// Check name with just hyphens replaced
// 同理替换中划线
String noHyphenName = name.replace('-', '_');
if (!name.equals(noHyphenName) && containsKey(noHyphenName)) {
return noHyphenName;
}
// Check name with dots and hyphens replaced
// 替换.后替换中划线,即2种同时存在
String noDotNoHyphenName = noDotName.replace('-', '_');
if (!noDotName.equals(noDotNoHyphenName) && containsKey(noDotNoHyphenName)) {
return noDotNoHyphenName;
}
// Give up 牛逼注释
return null;
}

private boolean containsKey(String name) {
return (isSecurityManagerPresent() ? this.source.keySet().contains(name) : this.source.containsKey(name));
}

至此,就知道了容器或者虚拟机注入环境变量,但是spring读取却可以使用常规xxx.xxx小写读取到的原理,Spring充当了转义器的角色。

 

总结
因为框架有定制,笔者一开始并不知道为啥能读取成功,而实际上却神奇的读取到环境变量了,翻遍定制代码都没发现哪里处理的,开始怀疑Spring干的,而Spring读取环境变量的代码就是org.springframework.core.env.SystemEnvironmentPropertySource,刚好就找到处理的代码了,明白原理。
————————————————

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。

原文链接:https://blog.csdn.net/fenglllle/article/details/126942480

标签:读取,return,String,Spring,Environment,环境变量,name
From: https://www.cnblogs.com/lbzwd/p/18258178

相关文章

  • 深入理解微服务架构 - Spring Cloud
    目录一SpringCloud概述什么是SpringCloud?为什么选择SpringCloud?二SpringCloud的核心组件SpringCloudConfig概述工作原理示例代码SpringCloudNetflixEureka概述工作原理示例代码SpringCloudRibbon概述工作原理示例代码SpringCloudHystrix概述......
  • Java项目-基于SpringCloud+springboot+vue的分布式架构网上商城系统(源码+数据库+文档
    源码获取:https://download.csdn.net/download/u011832806/89440647基于SpringCloud+SpringBoot+Vue的分布式架构网上商城系统   开发语言:Java   数据库:MySQL   技术:SpringCloud+SpringBoot+MyBatis+Vue.js+eureka   工具:IDEA/Ecilpse、Navicat、Maven经......
  • Java毕业设计-基于springboot开发的网上购物商城系统研发-毕业论文(附毕设源代码)
    文章目录前言一、毕设成果演示(源代码在文末)二、毕设摘要展示1、开发说明2、需求/流程分析3、系统功能结构三、系统实现展示1、用户功能模块的实现1.1用户注册界面1.2用户登录界面1.3个人中心界面1.4商品详情界面1.5购物车界面1.6我的订单界面1.7我的地址界面2、管理员......
  • Java毕业设计-基于springboot开发的网上租赁系统设计与实现-毕业论文(附毕设源代码)
    文章目录前言一、毕设成果演示(源代码在文末)二、毕设摘要展示1、开发说明2、需求/流程分析3、系统功能结构三、系统实现展示1、用户功能模块的实现1.1用户注册界面1.2用户登录界面1.3个人信息界面1.4商品详情界面1.5购物车界面1.6订单信息界面1.7发货订单界面2、管理员......
  • Java项目-基于springboot+vue的学习平台(源码+数据库+文档)​
    源码获取:https://download.csdn.net/download/u011832806/89456223基于SpringBoot+Vue的学习平台开发语言:Java数据库:MySQL技术:SpringBoot+MyBatis+Vue.js工具:IDEA/Ecilpse、Navicat、Maven在Internet高速发展的今天,我们生活的各个领域都涉及到计算机的应用,其中包括学习平台......
  • Bat中实现git和Maven环境变量检测,拉取git代码并打包maven项目,读取子目录差异性yml文件
    场景Docker+Jenkins+Pipline实现SpringBoot项目input选择不同差异性yml文件打包、执行sh打包压缩包、使用archiveArtifacts下载制品(jar包、压缩包):https://blog.csdn.net/BADAO_LIUMANG_QIZHI/article/details/139748758上面实现打包后台jar包并选择不同的差异性yml文件打包成......
  • SpringBoot+Selenium模拟用户操作浏览器
    SeleniumSelenium是一个用于Web应用程序自动化测试的开源工具套件。它主要用于以下目的:浏览器自动化:Selenium能够模拟真实用户在不同浏览器(如Chrome、Firefox、IE/Edge等)中的交互行为,通过编程方式控制浏览器执行一系列操作,例如点击按钮、填写表单、导航页面等。兼容性测试:通过......
  • 基于SpringBoot的冬奥会科普平台
    采用技术基于SpringBoot的冬奥会科普平台的设计与实现~开发语言:Java数据库:MySQL技术:SpringBoot+MyBatis工具:IDEA/Ecilpse、Navicat、Maven页面展示效果登录注册登录用户注册系统功能首页冬奥会运行页面精彩视频页面冬奥会论坛管理员页面首页用户管......
  • 最新SpringBoot项目篮球论坛系统
    采用技术最新SpringBoot项目篮球论坛系统的设计与实现~开发语言:Java数据库:MySQL技术:SpringBoot+MyBatis工具:IDEA/Ecilpse、Navicat、Maven页面展示效果系统功能首页用户注册篮球论坛个人中心管理员功能登录首页用户管理篮球论坛管理系统管理用......
  • spring中@Async注解底层线程池实现原理
    一、前言开发中我们经常会用到异步方法调用,具体到代码层面,异步方法调用的实现方式有很多种,比如最原始的通过实现Runnable接口或者继承Thread类创建异步线程,然后启动异步线程;再如,可以直接用java.util.concurrent包提供的线程池相关API实现异步方法调用。如果说可以用一行代码快速......